2010-04-29 14:14:44 +02:00
|
|
|
/*
|
|
|
|
* Virtio 9p backend
|
|
|
|
*
|
|
|
|
* Copyright IBM, Corp. 2010
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Anthony Liguori <aliguori@us.ibm.com>
|
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2. See
|
|
|
|
* the COPYING file in the top-level directory.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2011-06-01 09:05:14 +02:00
|
|
|
#include "hw/virtio.h"
|
|
|
|
#include "hw/pc.h"
|
2010-04-29 14:14:44 +02:00
|
|
|
#include "qemu_socket.h"
|
2011-06-01 09:05:14 +02:00
|
|
|
#include "hw/virtio-pci.h"
|
2010-04-29 14:14:44 +02:00
|
|
|
#include "virtio-9p.h"
|
|
|
|
#include "fsdev/qemu-fsdev.h"
|
2010-10-18 11:58:16 +02:00
|
|
|
#include "virtio-9p-xattr.h"
|
2011-05-18 23:18:05 +02:00
|
|
|
#include "virtio-9p-coth.h"
|
2011-10-12 15:41:25 +02:00
|
|
|
#include "trace.h"
|
2010-04-29 14:14:44 +02:00
|
|
|
|
2011-05-18 12:10:57 +02:00
|
|
|
int open_fd_hw;
|
|
|
|
int total_open_fd;
|
|
|
|
static int open_fd_rc;
|
2010-04-29 14:14:44 +02:00
|
|
|
|
2010-06-01 22:30:51 +02:00
|
|
|
enum {
|
|
|
|
Oread = 0x00,
|
|
|
|
Owrite = 0x01,
|
|
|
|
Ordwr = 0x02,
|
|
|
|
Oexec = 0x03,
|
|
|
|
Oexcl = 0x04,
|
|
|
|
Otrunc = 0x10,
|
|
|
|
Orexec = 0x20,
|
|
|
|
Orclose = 0x40,
|
|
|
|
Oappend = 0x80,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int omode_to_uflags(int8_t mode)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
switch (mode & 3) {
|
|
|
|
case Oread:
|
|
|
|
ret = O_RDONLY;
|
|
|
|
break;
|
|
|
|
case Ordwr:
|
|
|
|
ret = O_RDWR;
|
|
|
|
break;
|
|
|
|
case Owrite:
|
|
|
|
ret = O_WRONLY;
|
|
|
|
break;
|
|
|
|
case Oexec:
|
|
|
|
ret = O_RDONLY;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mode & Otrunc) {
|
|
|
|
ret |= O_TRUNC;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mode & Oappend) {
|
|
|
|
ret |= O_APPEND;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mode & Oexcl) {
|
|
|
|
ret |= O_EXCL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-10-12 15:41:24 +02:00
|
|
|
static int dotl_to_at_flags(int flags)
|
|
|
|
{
|
|
|
|
int rflags = 0;
|
|
|
|
if (flags & P9_DOTL_AT_REMOVEDIR) {
|
|
|
|
rflags |= AT_REMOVEDIR;
|
|
|
|
}
|
|
|
|
return rflags;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct dotl_openflag_map {
|
|
|
|
int dotl_flag;
|
|
|
|
int open_flag;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int dotl_to_open_flags(int flags)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
/*
|
|
|
|
* We have same bits for P9_DOTL_READONLY, P9_DOTL_WRONLY
|
|
|
|
* and P9_DOTL_NOACCESS
|
|
|
|
*/
|
|
|
|
int oflags = flags & O_ACCMODE;
|
|
|
|
|
|
|
|
struct dotl_openflag_map dotl_oflag_map[] = {
|
|
|
|
{ P9_DOTL_CREATE, O_CREAT },
|
|
|
|
{ P9_DOTL_EXCL, O_EXCL },
|
|
|
|
{ P9_DOTL_NOCTTY , O_NOCTTY },
|
|
|
|
{ P9_DOTL_TRUNC, O_TRUNC },
|
|
|
|
{ P9_DOTL_APPEND, O_APPEND },
|
|
|
|
{ P9_DOTL_NONBLOCK, O_NONBLOCK } ,
|
|
|
|
{ P9_DOTL_DSYNC, O_DSYNC },
|
|
|
|
{ P9_DOTL_FASYNC, FASYNC },
|
|
|
|
{ P9_DOTL_DIRECT, O_DIRECT },
|
|
|
|
{ P9_DOTL_LARGEFILE, O_LARGEFILE },
|
|
|
|
{ P9_DOTL_DIRECTORY, O_DIRECTORY },
|
|
|
|
{ P9_DOTL_NOFOLLOW, O_NOFOLLOW },
|
|
|
|
{ P9_DOTL_NOATIME, O_NOATIME },
|
|
|
|
{ P9_DOTL_SYNC, O_SYNC },
|
|
|
|
};
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(dotl_oflag_map); i++) {
|
|
|
|
if (flags & dotl_oflag_map[i].dotl_flag) {
|
|
|
|
oflags |= dotl_oflag_map[i].open_flag;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return oflags;
|
|
|
|
}
|
|
|
|
|
2010-06-14 22:34:41 +02:00
|
|
|
void cred_init(FsCred *credp)
|
2010-04-29 14:14:47 +02:00
|
|
|
{
|
2010-06-14 22:34:41 +02:00
|
|
|
credp->fc_uid = -1;
|
|
|
|
credp->fc_gid = -1;
|
|
|
|
credp->fc_mode = -1;
|
|
|
|
credp->fc_rdev = -1;
|
2010-04-29 14:14:47 +02:00
|
|
|
}
|
|
|
|
|
2011-10-12 15:41:23 +02:00
|
|
|
static int get_dotl_openflags(V9fsState *s, int oflags)
|
|
|
|
{
|
|
|
|
int flags;
|
|
|
|
/*
|
|
|
|
* Filter the client open flags
|
|
|
|
*/
|
2011-10-12 15:41:24 +02:00
|
|
|
flags = dotl_to_open_flags(oflags);
|
|
|
|
flags &= ~(O_NOCTTY | O_ASYNC | O_CREAT);
|
2011-10-12 15:41:23 +02:00
|
|
|
/*
|
|
|
|
* Ignore direct disk access hint until the server supports it.
|
|
|
|
*/
|
|
|
|
flags &= ~O_DIRECT;
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
2011-05-24 11:51:27 +02:00
|
|
|
void v9fs_string_init(V9fsString *str)
|
2010-04-29 14:14:46 +02:00
|
|
|
{
|
|
|
|
str->data = NULL;
|
|
|
|
str->size = 0;
|
|
|
|
}
|
|
|
|
|
2011-05-24 11:51:27 +02:00
|
|
|
void v9fs_string_free(V9fsString *str)
|
2010-04-29 14:14:46 +02:00
|
|
|
{
|
2011-08-21 05:09:37 +02:00
|
|
|
g_free(str->data);
|
2010-04-29 14:14:46 +02:00
|
|
|
str->data = NULL;
|
|
|
|
str->size = 0;
|
|
|
|
}
|
|
|
|
|
2011-05-24 11:51:27 +02:00
|
|
|
void v9fs_string_null(V9fsString *str)
|
2010-04-29 14:14:46 +02:00
|
|
|
{
|
|
|
|
v9fs_string_free(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int number_to_string(void *arg, char type)
|
|
|
|
{
|
|
|
|
unsigned int ret = 0;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case 'u': {
|
|
|
|
unsigned int num = *(unsigned int *)arg;
|
|
|
|
|
|
|
|
do {
|
|
|
|
ret++;
|
|
|
|
num = num/10;
|
|
|
|
} while (num);
|
|
|
|
break;
|
|
|
|
}
|
2010-10-07 02:09:42 +02:00
|
|
|
case 'U': {
|
|
|
|
unsigned long num = *(unsigned long *)arg;
|
|
|
|
do {
|
|
|
|
ret++;
|
|
|
|
num = num/10;
|
|
|
|
} while (num);
|
|
|
|
break;
|
|
|
|
}
|
2010-04-29 14:14:46 +02:00
|
|
|
default:
|
|
|
|
printf("Number_to_string: Unknown number format\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-09-27 18:45:47 +02:00
|
|
|
static int GCC_FMT_ATTR(2, 0)
|
|
|
|
v9fs_string_alloc_printf(char **strp, const char *fmt, va_list ap)
|
2010-04-29 14:14:46 +02:00
|
|
|
{
|
|
|
|
va_list ap2;
|
|
|
|
char *iter = (char *)fmt;
|
|
|
|
int len = 0;
|
|
|
|
int nr_args = 0;
|
|
|
|
char *arg_char_ptr;
|
|
|
|
unsigned int arg_uint;
|
2010-10-07 02:09:42 +02:00
|
|
|
unsigned long arg_ulong;
|
2010-04-29 14:14:46 +02:00
|
|
|
|
|
|
|
/* Find the number of %'s that denotes an argument */
|
|
|
|
for (iter = strstr(iter, "%"); iter; iter = strstr(iter, "%")) {
|
|
|
|
nr_args++;
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = strlen(fmt) - 2*nr_args;
|
|
|
|
|
|
|
|
if (!nr_args) {
|
|
|
|
goto alloc_print;
|
|
|
|
}
|
|
|
|
|
|
|
|
va_copy(ap2, ap);
|
|
|
|
|
|
|
|
iter = (char *)fmt;
|
|
|
|
|
|
|
|
/* Now parse the format string */
|
|
|
|
for (iter = strstr(iter, "%"); iter; iter = strstr(iter, "%")) {
|
|
|
|
iter++;
|
|
|
|
switch (*iter) {
|
|
|
|
case 'u':
|
|
|
|
arg_uint = va_arg(ap2, unsigned int);
|
|
|
|
len += number_to_string((void *)&arg_uint, 'u');
|
|
|
|
break;
|
2010-10-07 02:09:42 +02:00
|
|
|
case 'l':
|
|
|
|
if (*++iter == 'u') {
|
|
|
|
arg_ulong = va_arg(ap2, unsigned long);
|
|
|
|
len += number_to_string((void *)&arg_ulong, 'U');
|
|
|
|
} else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
2010-04-29 14:14:46 +02:00
|
|
|
case 's':
|
|
|
|
arg_char_ptr = va_arg(ap2, char *);
|
|
|
|
len += strlen(arg_char_ptr);
|
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
len += 1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
fprintf(stderr,
|
|
|
|
"v9fs_string_alloc_printf:Incorrect format %c", *iter);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
alloc_print:
|
2011-08-21 05:09:37 +02:00
|
|
|
*strp = g_malloc((len + 1) * sizeof(**strp));
|
2010-04-29 14:14:46 +02:00
|
|
|
|
|
|
|
return vsprintf(*strp, fmt, ap);
|
|
|
|
}
|
|
|
|
|
2011-05-24 11:51:27 +02:00
|
|
|
void GCC_FMT_ATTR(2, 3)
|
2010-09-27 18:45:47 +02:00
|
|
|
v9fs_string_sprintf(V9fsString *str, const char *fmt, ...)
|
2010-04-29 14:14:46 +02:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
v9fs_string_free(str);
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
err = v9fs_string_alloc_printf(&str->data, fmt, ap);
|
|
|
|
BUG_ON(err == -1);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
str->size = err;
|
|
|
|
}
|
|
|
|
|
2011-05-24 11:51:27 +02:00
|
|
|
void v9fs_string_copy(V9fsString *lhs, V9fsString *rhs)
|
2010-04-29 14:14:46 +02:00
|
|
|
{
|
|
|
|
v9fs_string_free(lhs);
|
|
|
|
v9fs_string_sprintf(lhs, "%s", rhs->data);
|
|
|
|
}
|
|
|
|
|
2011-09-09 11:44:18 +02:00
|
|
|
void v9fs_path_init(V9fsPath *path)
|
|
|
|
{
|
|
|
|
path->data = NULL;
|
|
|
|
path->size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void v9fs_path_free(V9fsPath *path)
|
|
|
|
{
|
|
|
|
g_free(path->data);
|
|
|
|
path->data = NULL;
|
|
|
|
path->size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void v9fs_path_copy(V9fsPath *lhs, V9fsPath *rhs)
|
|
|
|
{
|
|
|
|
v9fs_path_free(lhs);
|
|
|
|
lhs->data = g_malloc(rhs->size);
|
|
|
|
memcpy(lhs->data, rhs->data, rhs->size);
|
|
|
|
lhs->size = rhs->size;
|
|
|
|
}
|
|
|
|
|
|
|
|
int v9fs_name_to_path(V9fsState *s, V9fsPath *dirpath,
|
|
|
|
const char *name, V9fsPath *path)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
err = s->ops->name_to_path(&s->ctx, dirpath, name, path);
|
|
|
|
if (err < 0) {
|
|
|
|
err = -errno;
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2011-06-01 09:05:11 +02:00
|
|
|
/*
|
|
|
|
* Return TRUE if s1 is an ancestor of s2.
|
|
|
|
*
|
|
|
|
* E.g. "a/b" is an ancestor of "a/b/c" but not of "a/bc/d".
|
|
|
|
* As a special case, We treat s1 as ancestor of s2 if they are same!
|
|
|
|
*/
|
2011-09-09 11:44:18 +02:00
|
|
|
static int v9fs_path_is_ancestor(V9fsPath *s1, V9fsPath *s2)
|
2011-06-01 09:05:11 +02:00
|
|
|
{
|
2011-09-09 11:44:18 +02:00
|
|
|
if (!strncmp(s1->data, s2->data, s1->size - 1)) {
|
|
|
|
if (s2->data[s1->size - 1] == '\0' || s2->data[s1->size - 1] == '/') {
|
2011-06-01 09:05:11 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-29 14:14:46 +02:00
|
|
|
static size_t v9fs_string_size(V9fsString *str)
|
|
|
|
{
|
|
|
|
return str->size;
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:24 +02:00
|
|
|
/*
|
|
|
|
* returns 0 if fid got re-opened, 1 if not, < 0 on error */
|
2011-08-02 08:06:17 +02:00
|
|
|
static int v9fs_reopen_fid(V9fsPDU *pdu, V9fsFidState *f)
|
2011-08-02 08:06:24 +02:00
|
|
|
{
|
|
|
|
int err = 1;
|
|
|
|
if (f->fid_type == P9_FID_FILE) {
|
|
|
|
if (f->fs.fd == -1) {
|
|
|
|
do {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_open(pdu, f, f->open_flags);
|
|
|
|
} while (err == -EINTR && !pdu->cancelled);
|
2011-08-02 08:06:24 +02:00
|
|
|
}
|
|
|
|
} else if (f->fid_type == P9_FID_DIR) {
|
|
|
|
if (f->fs.dir == NULL) {
|
|
|
|
do {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_opendir(pdu, f);
|
|
|
|
} while (err == -EINTR && !pdu->cancelled);
|
2011-08-02 08:06:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static V9fsFidState *get_fid(V9fsPDU *pdu, int32_t fid)
|
2010-04-29 14:14:48 +02:00
|
|
|
{
|
2011-05-18 12:10:57 +02:00
|
|
|
int err;
|
2010-04-29 14:14:48 +02:00
|
|
|
V9fsFidState *f;
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:48 +02:00
|
|
|
|
|
|
|
for (f = s->fid_list; f; f = f->next) {
|
2011-05-18 14:08:07 +02:00
|
|
|
BUG_ON(f->clunked);
|
2010-04-29 14:14:48 +02:00
|
|
|
if (f->fid == fid) {
|
2011-05-18 12:10:57 +02:00
|
|
|
/*
|
|
|
|
* Update the fid ref upfront so that
|
|
|
|
* we don't get reclaimed when we yield
|
|
|
|
* in open later.
|
|
|
|
*/
|
2011-05-18 14:08:07 +02:00
|
|
|
f->ref++;
|
2011-05-18 12:10:57 +02:00
|
|
|
/*
|
|
|
|
* check whether we need to reopen the
|
|
|
|
* file. We might have closed the fd
|
|
|
|
* while trying to free up some file
|
|
|
|
* descriptors.
|
|
|
|
*/
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_reopen_fid(pdu, f);
|
2011-08-02 08:06:24 +02:00
|
|
|
if (err < 0) {
|
|
|
|
f->ref--;
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-05-18 12:10:57 +02:00
|
|
|
/*
|
|
|
|
* Mark the fid as referenced so that the LRU
|
|
|
|
* reclaim won't close the file descriptor
|
|
|
|
*/
|
|
|
|
f->flags |= FID_REFERENCED;
|
2010-04-29 14:14:48 +02:00
|
|
|
return f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static V9fsFidState *alloc_fid(V9fsState *s, int32_t fid)
|
|
|
|
{
|
|
|
|
V9fsFidState *f;
|
|
|
|
|
2011-05-18 14:08:07 +02:00
|
|
|
for (f = s->fid_list; f; f = f->next) {
|
|
|
|
/* If fid is already there return NULL */
|
|
|
|
BUG_ON(f->clunked);
|
|
|
|
if (f->fid == fid) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2010-04-29 14:14:48 +02:00
|
|
|
}
|
2011-08-21 05:09:37 +02:00
|
|
|
f = g_malloc0(sizeof(V9fsFidState));
|
2010-04-29 14:14:48 +02:00
|
|
|
f->fid = fid;
|
2010-09-02 07:39:06 +02:00
|
|
|
f->fid_type = P9_FID_NONE;
|
2011-05-18 14:08:07 +02:00
|
|
|
f->ref = 1;
|
2011-05-18 12:10:57 +02:00
|
|
|
/*
|
|
|
|
* Mark the fid as referenced so that the LRU
|
|
|
|
* reclaim won't close the file descriptor
|
|
|
|
*/
|
|
|
|
f->flags |= FID_REFERENCED;
|
2010-04-29 14:14:48 +02:00
|
|
|
f->next = s->fid_list;
|
|
|
|
s->fid_list = f;
|
|
|
|
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int v9fs_xattr_fid_clunk(V9fsPDU *pdu, V9fsFidState *fidp)
|
2010-09-02 07:39:07 +02:00
|
|
|
{
|
|
|
|
int retval = 0;
|
|
|
|
|
|
|
|
if (fidp->fs.xattr.copied_len == -1) {
|
|
|
|
/* getxattr/listxattr fid */
|
|
|
|
goto free_value;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* if this is fid for setxattr. clunk should
|
|
|
|
* result in setxattr localcall
|
|
|
|
*/
|
|
|
|
if (fidp->fs.xattr.len != fidp->fs.xattr.copied_len) {
|
|
|
|
/* clunk after partial write */
|
|
|
|
retval = -EINVAL;
|
|
|
|
goto free_out;
|
|
|
|
}
|
2010-08-26 07:45:23 +02:00
|
|
|
if (fidp->fs.xattr.len) {
|
2011-08-02 08:06:17 +02:00
|
|
|
retval = v9fs_co_lsetxattr(pdu, &fidp->path, &fidp->fs.xattr.name,
|
2010-08-26 07:45:23 +02:00
|
|
|
fidp->fs.xattr.value,
|
|
|
|
fidp->fs.xattr.len,
|
|
|
|
fidp->fs.xattr.flags);
|
|
|
|
} else {
|
2011-08-02 08:06:17 +02:00
|
|
|
retval = v9fs_co_lremovexattr(pdu, &fidp->path, &fidp->fs.xattr.name);
|
2010-08-26 07:45:23 +02:00
|
|
|
}
|
2010-09-02 07:39:07 +02:00
|
|
|
free_out:
|
|
|
|
v9fs_string_free(&fidp->fs.xattr.name);
|
|
|
|
free_value:
|
|
|
|
if (fidp->fs.xattr.value) {
|
2011-08-21 05:09:37 +02:00
|
|
|
g_free(fidp->fs.xattr.value);
|
2010-09-02 07:39:07 +02:00
|
|
|
}
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int free_fid(V9fsPDU *pdu, V9fsFidState *fidp)
|
2010-04-29 14:14:48 +02:00
|
|
|
{
|
2010-09-02 07:39:07 +02:00
|
|
|
int retval = 0;
|
2011-05-18 14:08:07 +02:00
|
|
|
|
|
|
|
if (fidp->fid_type == P9_FID_FILE) {
|
2011-05-18 12:10:57 +02:00
|
|
|
/* If we reclaimed the fd no need to close */
|
|
|
|
if (fidp->fs.fd != -1) {
|
2011-08-02 08:06:17 +02:00
|
|
|
retval = v9fs_co_close(pdu, fidp->fs.fd);
|
2011-05-18 12:10:57 +02:00
|
|
|
}
|
2011-05-18 14:08:07 +02:00
|
|
|
} else if (fidp->fid_type == P9_FID_DIR) {
|
2011-05-18 13:38:34 +02:00
|
|
|
if (fidp->fs.dir != NULL) {
|
2011-08-02 08:06:17 +02:00
|
|
|
retval = v9fs_co_closedir(pdu, fidp->fs.dir);
|
2011-05-18 13:38:34 +02:00
|
|
|
}
|
2011-05-18 14:08:07 +02:00
|
|
|
} else if (fidp->fid_type == P9_FID_XATTR) {
|
2011-08-02 08:06:17 +02:00
|
|
|
retval = v9fs_xattr_fid_clunk(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
}
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&fidp->path);
|
2011-05-18 14:08:07 +02:00
|
|
|
g_free(fidp);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static void put_fid(V9fsPDU *pdu, V9fsFidState *fidp)
|
2011-05-18 14:08:07 +02:00
|
|
|
{
|
|
|
|
BUG_ON(!fidp->ref);
|
|
|
|
fidp->ref--;
|
2011-05-18 12:10:57 +02:00
|
|
|
/*
|
|
|
|
* Don't free the fid if it is in reclaim list
|
|
|
|
*/
|
2011-05-18 14:08:07 +02:00
|
|
|
if (!fidp->ref && fidp->clunked) {
|
2011-08-02 08:06:17 +02:00
|
|
|
free_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:24 +02:00
|
|
|
static V9fsFidState *clunk_fid(V9fsState *s, int32_t fid)
|
2011-05-18 14:08:07 +02:00
|
|
|
{
|
2010-04-29 14:14:48 +02:00
|
|
|
V9fsFidState **fidpp, *fidp;
|
|
|
|
|
|
|
|
for (fidpp = &s->fid_list; *fidpp; fidpp = &(*fidpp)->next) {
|
|
|
|
if ((*fidpp)->fid == fid) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (*fidpp == NULL) {
|
2011-08-02 08:06:24 +02:00
|
|
|
return NULL;
|
2010-04-29 14:14:48 +02:00
|
|
|
}
|
|
|
|
fidp = *fidpp;
|
|
|
|
*fidpp = fidp->next;
|
2011-05-18 14:08:07 +02:00
|
|
|
fidp->clunked = 1;
|
2011-08-02 08:06:24 +02:00
|
|
|
return fidp;
|
2010-04-29 14:14:48 +02:00
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
void v9fs_reclaim_fd(V9fsPDU *pdu)
|
2011-05-18 12:10:57 +02:00
|
|
|
{
|
|
|
|
int reclaim_count = 0;
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsState *s = pdu->s;
|
2011-05-18 12:10:57 +02:00
|
|
|
V9fsFidState *f, *reclaim_list = NULL;
|
|
|
|
|
|
|
|
for (f = s->fid_list; f; f = f->next) {
|
|
|
|
/*
|
|
|
|
* Unlink fids cannot be reclaimed. Check
|
|
|
|
* for them and skip them. Also skip fids
|
|
|
|
* currently being operated on.
|
|
|
|
*/
|
|
|
|
if (f->ref || f->flags & FID_NON_RECLAIMABLE) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* if it is a recently referenced fid
|
|
|
|
* we leave the fid untouched and clear the
|
|
|
|
* reference bit. We come back to it later
|
|
|
|
* in the next iteration. (a simple LRU without
|
|
|
|
* moving list elements around)
|
|
|
|
*/
|
|
|
|
if (f->flags & FID_REFERENCED) {
|
|
|
|
f->flags &= ~FID_REFERENCED;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Add fids to reclaim list.
|
|
|
|
*/
|
|
|
|
if (f->fid_type == P9_FID_FILE) {
|
|
|
|
if (f->fs.fd != -1) {
|
|
|
|
/*
|
|
|
|
* Up the reference count so that
|
|
|
|
* a clunk request won't free this fid
|
|
|
|
*/
|
|
|
|
f->ref++;
|
|
|
|
f->rclm_lst = reclaim_list;
|
|
|
|
reclaim_list = f;
|
|
|
|
f->fs_reclaim.fd = f->fs.fd;
|
|
|
|
f->fs.fd = -1;
|
|
|
|
reclaim_count++;
|
|
|
|
}
|
2011-05-18 13:38:34 +02:00
|
|
|
} else if (f->fid_type == P9_FID_DIR) {
|
|
|
|
if (f->fs.dir != NULL) {
|
|
|
|
/*
|
|
|
|
* Up the reference count so that
|
|
|
|
* a clunk request won't free this fid
|
|
|
|
*/
|
|
|
|
f->ref++;
|
|
|
|
f->rclm_lst = reclaim_list;
|
|
|
|
reclaim_list = f;
|
|
|
|
f->fs_reclaim.dir = f->fs.dir;
|
|
|
|
f->fs.dir = NULL;
|
|
|
|
reclaim_count++;
|
|
|
|
}
|
2011-05-18 12:10:57 +02:00
|
|
|
}
|
|
|
|
if (reclaim_count >= open_fd_rc) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Now close the fid in reclaim list. Free them if they
|
|
|
|
* are already clunked.
|
|
|
|
*/
|
|
|
|
while (reclaim_list) {
|
|
|
|
f = reclaim_list;
|
|
|
|
reclaim_list = f->rclm_lst;
|
|
|
|
if (f->fid_type == P9_FID_FILE) {
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_close(pdu, f->fs_reclaim.fd);
|
2011-05-18 13:38:34 +02:00
|
|
|
} else if (f->fid_type == P9_FID_DIR) {
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_closedir(pdu, f->fs_reclaim.dir);
|
2011-05-18 12:10:57 +02:00
|
|
|
}
|
|
|
|
f->rclm_lst = NULL;
|
|
|
|
/*
|
|
|
|
* Now drop the fid reference, free it
|
|
|
|
* if clunked.
|
|
|
|
*/
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, f);
|
2011-05-18 12:10:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int v9fs_mark_fids_unreclaim(V9fsPDU *pdu, V9fsPath *path)
|
2011-05-18 12:10:57 +02:00
|
|
|
{
|
|
|
|
int err;
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsState *s = pdu->s;
|
2011-05-18 12:10:57 +02:00
|
|
|
V9fsFidState *fidp, head_fid;
|
|
|
|
|
|
|
|
head_fid.next = s->fid_list;
|
|
|
|
for (fidp = s->fid_list; fidp; fidp = fidp->next) {
|
2011-09-09 11:44:18 +02:00
|
|
|
if (fidp->path.size != path->size) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!memcmp(fidp->path.data, path->data, path->size)) {
|
2011-05-18 12:10:57 +02:00
|
|
|
/* Mark the fid non reclaimable. */
|
|
|
|
fidp->flags |= FID_NON_RECLAIMABLE;
|
2011-08-02 08:06:24 +02:00
|
|
|
|
|
|
|
/* reopen the file/dir if already closed */
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_reopen_fid(pdu, fidp);
|
2011-08-02 08:06:24 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Go back to head of fid list because
|
|
|
|
* the list could have got updated when
|
|
|
|
* switched to the worker thread
|
|
|
|
*/
|
|
|
|
if (err == 0) {
|
2011-05-18 12:10:57 +02:00
|
|
|
fidp = &head_fid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-29 14:14:48 +02:00
|
|
|
#define P9_QID_TYPE_DIR 0x80
|
|
|
|
#define P9_QID_TYPE_SYMLINK 0x02
|
|
|
|
|
|
|
|
#define P9_STAT_MODE_DIR 0x80000000
|
|
|
|
#define P9_STAT_MODE_APPEND 0x40000000
|
|
|
|
#define P9_STAT_MODE_EXCL 0x20000000
|
|
|
|
#define P9_STAT_MODE_MOUNT 0x10000000
|
|
|
|
#define P9_STAT_MODE_AUTH 0x08000000
|
|
|
|
#define P9_STAT_MODE_TMP 0x04000000
|
|
|
|
#define P9_STAT_MODE_SYMLINK 0x02000000
|
|
|
|
#define P9_STAT_MODE_LINK 0x01000000
|
|
|
|
#define P9_STAT_MODE_DEVICE 0x00800000
|
|
|
|
#define P9_STAT_MODE_NAMED_PIPE 0x00200000
|
|
|
|
#define P9_STAT_MODE_SOCKET 0x00100000
|
|
|
|
#define P9_STAT_MODE_SETUID 0x00080000
|
|
|
|
#define P9_STAT_MODE_SETGID 0x00040000
|
|
|
|
#define P9_STAT_MODE_SETVTX 0x00010000
|
|
|
|
|
|
|
|
#define P9_STAT_MODE_TYPE_BITS (P9_STAT_MODE_DIR | \
|
|
|
|
P9_STAT_MODE_SYMLINK | \
|
|
|
|
P9_STAT_MODE_LINK | \
|
|
|
|
P9_STAT_MODE_DEVICE | \
|
|
|
|
P9_STAT_MODE_NAMED_PIPE | \
|
|
|
|
P9_STAT_MODE_SOCKET)
|
|
|
|
|
|
|
|
/* This is the algorithm from ufs in spfs */
|
|
|
|
static void stat_to_qid(const struct stat *stbuf, V9fsQID *qidp)
|
|
|
|
{
|
|
|
|
size_t size;
|
|
|
|
|
2011-08-02 08:06:24 +02:00
|
|
|
memset(&qidp->path, 0, sizeof(qidp->path));
|
2010-04-29 14:14:48 +02:00
|
|
|
size = MIN(sizeof(stbuf->st_ino), sizeof(qidp->path));
|
|
|
|
memcpy(&qidp->path, &stbuf->st_ino, size);
|
|
|
|
qidp->version = stbuf->st_mtime ^ (stbuf->st_size << 8);
|
|
|
|
qidp->type = 0;
|
|
|
|
if (S_ISDIR(stbuf->st_mode)) {
|
|
|
|
qidp->type |= P9_QID_TYPE_DIR;
|
|
|
|
}
|
|
|
|
if (S_ISLNK(stbuf->st_mode)) {
|
|
|
|
qidp->type |= P9_QID_TYPE_SYMLINK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int fid_to_qid(V9fsPDU *pdu, V9fsFidState *fidp, V9fsQID *qidp)
|
2010-04-29 14:14:48 +02:00
|
|
|
{
|
|
|
|
struct stat stbuf;
|
|
|
|
int err;
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lstat(pdu, &fidp->path, &stbuf);
|
2011-05-08 09:45:29 +02:00
|
|
|
if (err < 0) {
|
2010-04-29 14:14:48 +02:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
stat_to_qid(&stbuf, qidp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-29 14:14:44 +02:00
|
|
|
static V9fsPDU *alloc_pdu(V9fsState *s)
|
|
|
|
{
|
|
|
|
V9fsPDU *pdu = NULL;
|
|
|
|
|
|
|
|
if (!QLIST_EMPTY(&s->free_list)) {
|
2011-08-02 08:06:17 +02:00
|
|
|
pdu = QLIST_FIRST(&s->free_list);
|
|
|
|
QLIST_REMOVE(pdu, next);
|
|
|
|
QLIST_INSERT_HEAD(&s->active_list, pdu, next);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
return pdu;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void free_pdu(V9fsState *s, V9fsPDU *pdu)
|
|
|
|
{
|
|
|
|
if (pdu) {
|
2011-08-02 08:06:17 +02:00
|
|
|
/*
|
|
|
|
* Cancelled pdu are added back to the freelist
|
|
|
|
* by flush request .
|
|
|
|
*/
|
|
|
|
if (!pdu->cancelled) {
|
|
|
|
QLIST_REMOVE(pdu, next);
|
|
|
|
QLIST_INSERT_HEAD(&s->free_list, pdu, next);
|
|
|
|
}
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t pdu_packunpack(void *addr, struct iovec *sg, int sg_count,
|
|
|
|
size_t offset, size_t size, int pack)
|
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
size_t copied = 0;
|
|
|
|
|
|
|
|
for (i = 0; size && i < sg_count; i++) {
|
|
|
|
size_t len;
|
|
|
|
if (offset >= sg[i].iov_len) {
|
|
|
|
/* skip this sg */
|
|
|
|
offset -= sg[i].iov_len;
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
len = MIN(sg[i].iov_len - offset, size);
|
|
|
|
if (pack) {
|
|
|
|
memcpy(sg[i].iov_base + offset, addr, len);
|
|
|
|
} else {
|
|
|
|
memcpy(addr, sg[i].iov_base + offset, len);
|
|
|
|
}
|
|
|
|
size -= len;
|
|
|
|
copied += len;
|
|
|
|
addr += len;
|
|
|
|
if (size) {
|
|
|
|
offset = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return copied;
|
|
|
|
}
|
|
|
|
|
2010-04-29 14:14:45 +02:00
|
|
|
static size_t pdu_unpack(void *dst, V9fsPDU *pdu, size_t offset, size_t size)
|
|
|
|
{
|
|
|
|
return pdu_packunpack(dst, pdu->elem.out_sg, pdu->elem.out_num,
|
|
|
|
offset, size, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t pdu_pack(V9fsPDU *pdu, size_t offset, const void *src,
|
|
|
|
size_t size)
|
|
|
|
{
|
|
|
|
return pdu_packunpack((void *)src, pdu->elem.in_sg, pdu->elem.in_num,
|
|
|
|
offset, size, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pdu_copy_sg(V9fsPDU *pdu, size_t offset, int rx, struct iovec *sg)
|
|
|
|
{
|
|
|
|
size_t pos = 0;
|
|
|
|
int i, j;
|
|
|
|
struct iovec *src_sg;
|
|
|
|
unsigned int num;
|
|
|
|
|
|
|
|
if (rx) {
|
|
|
|
src_sg = pdu->elem.in_sg;
|
|
|
|
num = pdu->elem.in_num;
|
|
|
|
} else {
|
|
|
|
src_sg = pdu->elem.out_sg;
|
|
|
|
num = pdu->elem.out_num;
|
|
|
|
}
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
for (i = 0; i < num; i++) {
|
|
|
|
if (offset <= pos) {
|
|
|
|
sg[j].iov_base = src_sg[i].iov_base;
|
|
|
|
sg[j].iov_len = src_sg[i].iov_len;
|
|
|
|
j++;
|
|
|
|
} else if (offset < (src_sg[i].iov_len + pos)) {
|
|
|
|
sg[j].iov_base = src_sg[i].iov_base;
|
|
|
|
sg[j].iov_len = src_sg[i].iov_len;
|
|
|
|
sg[j].iov_base += (offset - pos);
|
|
|
|
sg[j].iov_len -= (offset - pos);
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
pos += src_sg[i].iov_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return j;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t pdu_unmarshal(V9fsPDU *pdu, size_t offset, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
size_t old_offset = offset;
|
|
|
|
va_list ap;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
for (i = 0; fmt[i]; i++) {
|
|
|
|
switch (fmt[i]) {
|
|
|
|
case 'b': {
|
|
|
|
uint8_t *valp = va_arg(ap, uint8_t *);
|
|
|
|
offset += pdu_unpack(valp, pdu, offset, sizeof(*valp));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'w': {
|
|
|
|
uint16_t val, *valp;
|
|
|
|
valp = va_arg(ap, uint16_t *);
|
|
|
|
offset += pdu_unpack(&val, pdu, offset, sizeof(val));
|
2011-04-07 05:02:03 +02:00
|
|
|
*valp = le16_to_cpu(val);
|
2010-04-29 14:14:45 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'd': {
|
|
|
|
uint32_t val, *valp;
|
|
|
|
valp = va_arg(ap, uint32_t *);
|
|
|
|
offset += pdu_unpack(&val, pdu, offset, sizeof(val));
|
2011-04-07 05:02:03 +02:00
|
|
|
*valp = le32_to_cpu(val);
|
2010-04-29 14:14:45 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'q': {
|
|
|
|
uint64_t val, *valp;
|
|
|
|
valp = va_arg(ap, uint64_t *);
|
|
|
|
offset += pdu_unpack(&val, pdu, offset, sizeof(val));
|
2011-04-07 05:02:03 +02:00
|
|
|
*valp = le64_to_cpu(val);
|
2010-04-29 14:14:45 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'v': {
|
|
|
|
struct iovec *iov = va_arg(ap, struct iovec *);
|
|
|
|
int *iovcnt = va_arg(ap, int *);
|
|
|
|
*iovcnt = pdu_copy_sg(pdu, offset, 0, iov);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 's': {
|
|
|
|
V9fsString *str = va_arg(ap, V9fsString *);
|
|
|
|
offset += pdu_unmarshal(pdu, offset, "w", &str->size);
|
|
|
|
/* FIXME: sanity check str->size */
|
2011-08-21 05:09:37 +02:00
|
|
|
str->data = g_malloc(str->size + 1);
|
2010-04-29 14:14:45 +02:00
|
|
|
offset += pdu_unpack(str->data, pdu, offset, str->size);
|
|
|
|
str->data[str->size] = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'Q': {
|
|
|
|
V9fsQID *qidp = va_arg(ap, V9fsQID *);
|
|
|
|
offset += pdu_unmarshal(pdu, offset, "bdq",
|
|
|
|
&qidp->type, &qidp->version, &qidp->path);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'S': {
|
|
|
|
V9fsStat *statp = va_arg(ap, V9fsStat *);
|
|
|
|
offset += pdu_unmarshal(pdu, offset, "wwdQdddqsssssddd",
|
|
|
|
&statp->size, &statp->type, &statp->dev,
|
|
|
|
&statp->qid, &statp->mode, &statp->atime,
|
|
|
|
&statp->mtime, &statp->length,
|
|
|
|
&statp->name, &statp->uid, &statp->gid,
|
|
|
|
&statp->muid, &statp->extension,
|
|
|
|
&statp->n_uid, &statp->n_gid,
|
|
|
|
&statp->n_muid);
|
|
|
|
break;
|
|
|
|
}
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
case 'I': {
|
|
|
|
V9fsIattr *iattr = va_arg(ap, V9fsIattr *);
|
|
|
|
offset += pdu_unmarshal(pdu, offset, "ddddqqqqq",
|
|
|
|
&iattr->valid, &iattr->mode,
|
|
|
|
&iattr->uid, &iattr->gid, &iattr->size,
|
|
|
|
&iattr->atime_sec, &iattr->atime_nsec,
|
|
|
|
&iattr->mtime_sec, &iattr->mtime_nsec);
|
|
|
|
break;
|
|
|
|
}
|
2010-04-29 14:14:45 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
return offset - old_offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t pdu_marshal(V9fsPDU *pdu, size_t offset, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
size_t old_offset = offset;
|
|
|
|
va_list ap;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
for (i = 0; fmt[i]; i++) {
|
|
|
|
switch (fmt[i]) {
|
|
|
|
case 'b': {
|
|
|
|
uint8_t val = va_arg(ap, int);
|
|
|
|
offset += pdu_pack(pdu, offset, &val, sizeof(val));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'w': {
|
|
|
|
uint16_t val;
|
|
|
|
cpu_to_le16w(&val, va_arg(ap, int));
|
|
|
|
offset += pdu_pack(pdu, offset, &val, sizeof(val));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'd': {
|
|
|
|
uint32_t val;
|
|
|
|
cpu_to_le32w(&val, va_arg(ap, uint32_t));
|
|
|
|
offset += pdu_pack(pdu, offset, &val, sizeof(val));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'q': {
|
|
|
|
uint64_t val;
|
|
|
|
cpu_to_le64w(&val, va_arg(ap, uint64_t));
|
|
|
|
offset += pdu_pack(pdu, offset, &val, sizeof(val));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'v': {
|
|
|
|
struct iovec *iov = va_arg(ap, struct iovec *);
|
|
|
|
int *iovcnt = va_arg(ap, int *);
|
|
|
|
*iovcnt = pdu_copy_sg(pdu, offset, 1, iov);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 's': {
|
|
|
|
V9fsString *str = va_arg(ap, V9fsString *);
|
|
|
|
offset += pdu_marshal(pdu, offset, "w", str->size);
|
|
|
|
offset += pdu_pack(pdu, offset, str->data, str->size);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'Q': {
|
|
|
|
V9fsQID *qidp = va_arg(ap, V9fsQID *);
|
|
|
|
offset += pdu_marshal(pdu, offset, "bdq",
|
|
|
|
qidp->type, qidp->version, qidp->path);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'S': {
|
|
|
|
V9fsStat *statp = va_arg(ap, V9fsStat *);
|
|
|
|
offset += pdu_marshal(pdu, offset, "wwdQdddqsssssddd",
|
|
|
|
statp->size, statp->type, statp->dev,
|
|
|
|
&statp->qid, statp->mode, statp->atime,
|
|
|
|
statp->mtime, statp->length, &statp->name,
|
|
|
|
&statp->uid, &statp->gid, &statp->muid,
|
|
|
|
&statp->extension, statp->n_uid,
|
|
|
|
statp->n_gid, statp->n_muid);
|
|
|
|
break;
|
|
|
|
}
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
case 'A': {
|
|
|
|
V9fsStatDotl *statp = va_arg(ap, V9fsStatDotl *);
|
|
|
|
offset += pdu_marshal(pdu, offset, "qQdddqqqqqqqqqqqqqqq",
|
|
|
|
statp->st_result_mask,
|
|
|
|
&statp->qid, statp->st_mode,
|
|
|
|
statp->st_uid, statp->st_gid,
|
|
|
|
statp->st_nlink, statp->st_rdev,
|
|
|
|
statp->st_size, statp->st_blksize, statp->st_blocks,
|
|
|
|
statp->st_atime_sec, statp->st_atime_nsec,
|
|
|
|
statp->st_mtime_sec, statp->st_mtime_nsec,
|
|
|
|
statp->st_ctime_sec, statp->st_ctime_nsec,
|
|
|
|
statp->st_btime_sec, statp->st_btime_nsec,
|
|
|
|
statp->st_gen, statp->st_data_version);
|
|
|
|
break;
|
|
|
|
}
|
2010-04-29 14:14:45 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
return offset - old_offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void complete_pdu(V9fsState *s, V9fsPDU *pdu, ssize_t len)
|
|
|
|
{
|
|
|
|
int8_t id = pdu->id + 1; /* Response */
|
|
|
|
|
|
|
|
if (len < 0) {
|
|
|
|
int err = -len;
|
2010-07-28 10:40:22 +02:00
|
|
|
len = 7;
|
2010-04-29 14:14:45 +02:00
|
|
|
|
2010-07-28 10:40:22 +02:00
|
|
|
if (s->proto_version != V9FS_PROTO_2000L) {
|
|
|
|
V9fsString str;
|
|
|
|
|
|
|
|
str.data = strerror(err);
|
|
|
|
str.size = strlen(str.data);
|
|
|
|
|
|
|
|
len += pdu_marshal(pdu, len, "s", &str);
|
|
|
|
id = P9_RERROR;
|
|
|
|
}
|
2010-04-29 14:14:45 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
len += pdu_marshal(pdu, len, "d", err);
|
2010-04-29 14:14:45 +02:00
|
|
|
|
2010-07-28 10:40:22 +02:00
|
|
|
if (s->proto_version == V9FS_PROTO_2000L) {
|
|
|
|
id = P9_RLERROR;
|
|
|
|
}
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_complete_pdu(pdu->tag, pdu->id, err); /* Trace ERROR */
|
2010-04-29 14:14:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* fill out the header */
|
|
|
|
pdu_marshal(pdu, 0, "dbw", (int32_t)len, id, pdu->tag);
|
|
|
|
|
|
|
|
/* keep these in sync */
|
|
|
|
pdu->size = len;
|
|
|
|
pdu->id = id;
|
|
|
|
|
|
|
|
/* push onto queue and notify */
|
|
|
|
virtqueue_push(s->vq, &pdu->elem, len);
|
|
|
|
|
|
|
|
/* FIXME: we should batch these completions */
|
|
|
|
virtio_notify(&s->vdev, s->vq);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
/* Now wakeup anybody waiting in flush for this request */
|
|
|
|
qemu_co_queue_next(&pdu->complete);
|
|
|
|
|
2010-04-29 14:14:45 +02:00
|
|
|
free_pdu(s, pdu);
|
|
|
|
}
|
|
|
|
|
2010-04-29 14:14:49 +02:00
|
|
|
static mode_t v9mode_to_mode(uint32_t mode, V9fsString *extension)
|
|
|
|
{
|
|
|
|
mode_t ret;
|
|
|
|
|
|
|
|
ret = mode & 0777;
|
|
|
|
if (mode & P9_STAT_MODE_DIR) {
|
|
|
|
ret |= S_IFDIR;
|
|
|
|
}
|
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (mode & P9_STAT_MODE_SYMLINK) {
|
|
|
|
ret |= S_IFLNK;
|
|
|
|
}
|
|
|
|
if (mode & P9_STAT_MODE_SOCKET) {
|
|
|
|
ret |= S_IFSOCK;
|
|
|
|
}
|
|
|
|
if (mode & P9_STAT_MODE_NAMED_PIPE) {
|
|
|
|
ret |= S_IFIFO;
|
|
|
|
}
|
|
|
|
if (mode & P9_STAT_MODE_DEVICE) {
|
|
|
|
if (extension && extension->data[0] == 'c') {
|
|
|
|
ret |= S_IFCHR;
|
|
|
|
} else {
|
|
|
|
ret |= S_IFBLK;
|
2010-04-29 14:14:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(ret&~0777)) {
|
|
|
|
ret |= S_IFREG;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mode & P9_STAT_MODE_SETUID) {
|
|
|
|
ret |= S_ISUID;
|
|
|
|
}
|
|
|
|
if (mode & P9_STAT_MODE_SETGID) {
|
|
|
|
ret |= S_ISGID;
|
|
|
|
}
|
|
|
|
if (mode & P9_STAT_MODE_SETVTX) {
|
|
|
|
ret |= S_ISVTX;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int donttouch_stat(V9fsStat *stat)
|
|
|
|
{
|
|
|
|
if (stat->type == -1 &&
|
|
|
|
stat->dev == -1 &&
|
|
|
|
stat->qid.type == -1 &&
|
|
|
|
stat->qid.version == -1 &&
|
|
|
|
stat->qid.path == -1 &&
|
|
|
|
stat->mode == -1 &&
|
|
|
|
stat->atime == -1 &&
|
|
|
|
stat->mtime == -1 &&
|
|
|
|
stat->length == -1 &&
|
|
|
|
!stat->name.size &&
|
|
|
|
!stat->uid.size &&
|
|
|
|
!stat->gid.size &&
|
|
|
|
!stat->muid.size &&
|
|
|
|
stat->n_uid == -1 &&
|
|
|
|
stat->n_gid == -1 &&
|
|
|
|
stat->n_muid == -1) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void v9fs_stat_free(V9fsStat *stat)
|
|
|
|
{
|
|
|
|
v9fs_string_free(&stat->name);
|
|
|
|
v9fs_string_free(&stat->uid);
|
|
|
|
v9fs_string_free(&stat->gid);
|
|
|
|
v9fs_string_free(&stat->muid);
|
|
|
|
v9fs_string_free(&stat->extension);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t stat_to_v9mode(const struct stat *stbuf)
|
|
|
|
{
|
|
|
|
uint32_t mode;
|
|
|
|
|
|
|
|
mode = stbuf->st_mode & 0777;
|
|
|
|
if (S_ISDIR(stbuf->st_mode)) {
|
|
|
|
mode |= P9_STAT_MODE_DIR;
|
|
|
|
}
|
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (S_ISLNK(stbuf->st_mode)) {
|
|
|
|
mode |= P9_STAT_MODE_SYMLINK;
|
|
|
|
}
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (S_ISSOCK(stbuf->st_mode)) {
|
|
|
|
mode |= P9_STAT_MODE_SOCKET;
|
|
|
|
}
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (S_ISFIFO(stbuf->st_mode)) {
|
|
|
|
mode |= P9_STAT_MODE_NAMED_PIPE;
|
|
|
|
}
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (S_ISBLK(stbuf->st_mode) || S_ISCHR(stbuf->st_mode)) {
|
|
|
|
mode |= P9_STAT_MODE_DEVICE;
|
|
|
|
}
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (stbuf->st_mode & S_ISUID) {
|
|
|
|
mode |= P9_STAT_MODE_SETUID;
|
|
|
|
}
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (stbuf->st_mode & S_ISGID) {
|
|
|
|
mode |= P9_STAT_MODE_SETGID;
|
|
|
|
}
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (stbuf->st_mode & S_ISVTX) {
|
|
|
|
mode |= P9_STAT_MODE_SETVTX;
|
2010-04-29 14:14:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return mode;
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int stat_to_v9stat(V9fsPDU *pdu, V9fsPath *name,
|
2010-04-29 14:14:49 +02:00
|
|
|
const struct stat *stbuf,
|
|
|
|
V9fsStat *v9stat)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
const char *str;
|
|
|
|
|
|
|
|
memset(v9stat, 0, sizeof(*v9stat));
|
|
|
|
|
|
|
|
stat_to_qid(stbuf, &v9stat->qid);
|
|
|
|
v9stat->mode = stat_to_v9mode(stbuf);
|
|
|
|
v9stat->atime = stbuf->st_atime;
|
|
|
|
v9stat->mtime = stbuf->st_mtime;
|
|
|
|
v9stat->length = stbuf->st_size;
|
|
|
|
|
|
|
|
v9fs_string_null(&v9stat->uid);
|
|
|
|
v9fs_string_null(&v9stat->gid);
|
|
|
|
v9fs_string_null(&v9stat->muid);
|
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
v9stat->n_uid = stbuf->st_uid;
|
|
|
|
v9stat->n_gid = stbuf->st_gid;
|
|
|
|
v9stat->n_muid = 0;
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
v9fs_string_null(&v9stat->extension);
|
2010-04-29 14:14:49 +02:00
|
|
|
|
2010-07-28 10:25:05 +02:00
|
|
|
if (v9stat->mode & P9_STAT_MODE_SYMLINK) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_readlink(pdu, name, &v9stat->extension);
|
2011-08-08 20:06:41 +02:00
|
|
|
if (err < 0) {
|
2010-07-28 10:25:05 +02:00
|
|
|
return err;
|
2010-04-29 14:14:49 +02:00
|
|
|
}
|
2010-07-28 10:25:05 +02:00
|
|
|
} else if (v9stat->mode & P9_STAT_MODE_DEVICE) {
|
|
|
|
v9fs_string_sprintf(&v9stat->extension, "%c %u %u",
|
|
|
|
S_ISCHR(stbuf->st_mode) ? 'c' : 'b',
|
|
|
|
major(stbuf->st_rdev), minor(stbuf->st_rdev));
|
|
|
|
} else if (S_ISDIR(stbuf->st_mode) || S_ISREG(stbuf->st_mode)) {
|
2010-09-27 18:45:47 +02:00
|
|
|
v9fs_string_sprintf(&v9stat->extension, "%s %lu",
|
|
|
|
"HARDLINKCOUNT", (unsigned long)stbuf->st_nlink);
|
2010-04-29 14:14:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
str = strrchr(name->data, '/');
|
|
|
|
if (str) {
|
|
|
|
str += 1;
|
|
|
|
} else {
|
|
|
|
str = name->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
v9fs_string_sprintf(&v9stat->name, "%s", str);
|
|
|
|
|
|
|
|
v9stat->size = 61 +
|
|
|
|
v9fs_string_size(&v9stat->name) +
|
|
|
|
v9fs_string_size(&v9stat->uid) +
|
|
|
|
v9fs_string_size(&v9stat->gid) +
|
|
|
|
v9fs_string_size(&v9stat->muid) +
|
|
|
|
v9fs_string_size(&v9stat->extension);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
#define P9_STATS_MODE 0x00000001ULL
|
|
|
|
#define P9_STATS_NLINK 0x00000002ULL
|
|
|
|
#define P9_STATS_UID 0x00000004ULL
|
|
|
|
#define P9_STATS_GID 0x00000008ULL
|
|
|
|
#define P9_STATS_RDEV 0x00000010ULL
|
|
|
|
#define P9_STATS_ATIME 0x00000020ULL
|
|
|
|
#define P9_STATS_MTIME 0x00000040ULL
|
|
|
|
#define P9_STATS_CTIME 0x00000080ULL
|
|
|
|
#define P9_STATS_INO 0x00000100ULL
|
|
|
|
#define P9_STATS_SIZE 0x00000200ULL
|
|
|
|
#define P9_STATS_BLOCKS 0x00000400ULL
|
|
|
|
|
|
|
|
#define P9_STATS_BTIME 0x00000800ULL
|
|
|
|
#define P9_STATS_GEN 0x00001000ULL
|
|
|
|
#define P9_STATS_DATA_VERSION 0x00002000ULL
|
|
|
|
|
|
|
|
#define P9_STATS_BASIC 0x000007ffULL /* Mask for fields up to BLOCKS */
|
|
|
|
#define P9_STATS_ALL 0x00003fffULL /* Mask for All fields above */
|
|
|
|
|
|
|
|
|
|
|
|
static void stat_to_v9stat_dotl(V9fsState *s, const struct stat *stbuf,
|
2011-05-19 01:04:33 +02:00
|
|
|
V9fsStatDotl *v9lstat)
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
{
|
|
|
|
memset(v9lstat, 0, sizeof(*v9lstat));
|
|
|
|
|
|
|
|
v9lstat->st_mode = stbuf->st_mode;
|
|
|
|
v9lstat->st_nlink = stbuf->st_nlink;
|
|
|
|
v9lstat->st_uid = stbuf->st_uid;
|
|
|
|
v9lstat->st_gid = stbuf->st_gid;
|
|
|
|
v9lstat->st_rdev = stbuf->st_rdev;
|
|
|
|
v9lstat->st_size = stbuf->st_size;
|
|
|
|
v9lstat->st_blksize = stbuf->st_blksize;
|
|
|
|
v9lstat->st_blocks = stbuf->st_blocks;
|
|
|
|
v9lstat->st_atime_sec = stbuf->st_atime;
|
|
|
|
v9lstat->st_atime_nsec = stbuf->st_atim.tv_nsec;
|
|
|
|
v9lstat->st_mtime_sec = stbuf->st_mtime;
|
|
|
|
v9lstat->st_mtime_nsec = stbuf->st_mtim.tv_nsec;
|
|
|
|
v9lstat->st_ctime_sec = stbuf->st_ctime;
|
|
|
|
v9lstat->st_ctime_nsec = stbuf->st_ctim.tv_nsec;
|
|
|
|
/* Currently we only support BASIC fields in stat */
|
|
|
|
v9lstat->st_result_mask = P9_STATS_BASIC;
|
|
|
|
|
|
|
|
stat_to_qid(stbuf, &v9lstat->qid);
|
|
|
|
}
|
|
|
|
|
2010-04-29 14:14:50 +02:00
|
|
|
static struct iovec *adjust_sg(struct iovec *sg, int len, int *iovcnt)
|
|
|
|
{
|
|
|
|
while (len && *iovcnt) {
|
|
|
|
if (len < sg->iov_len) {
|
|
|
|
sg->iov_len -= len;
|
|
|
|
sg->iov_base += len;
|
|
|
|
len = 0;
|
|
|
|
} else {
|
|
|
|
len -= sg->iov_len;
|
|
|
|
sg++;
|
|
|
|
*iovcnt -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return sg;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct iovec *cap_sg(struct iovec *sg, int cap, int *cnt)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int total = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < *cnt; i++) {
|
|
|
|
if ((total + sg[i].iov_len) > cap) {
|
|
|
|
sg[i].iov_len -= ((total + sg[i].iov_len) - cap);
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
total += sg[i].iov_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cnt = i;
|
|
|
|
|
|
|
|
return sg;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void print_sg(struct iovec *sg, int cnt)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
printf("sg[%d]: {", cnt);
|
|
|
|
for (i = 0; i < cnt; i++) {
|
|
|
|
if (i) {
|
|
|
|
printf(", ");
|
|
|
|
}
|
|
|
|
printf("(%p, %zd)", sg[i].iov_base, sg[i].iov_len);
|
|
|
|
}
|
|
|
|
printf("}\n");
|
|
|
|
}
|
|
|
|
|
2011-09-09 11:44:18 +02:00
|
|
|
/* Will call this only for path name based fid */
|
|
|
|
static void v9fs_fix_path(V9fsPath *dst, V9fsPath *src, int len)
|
2010-04-29 14:15:00 +02:00
|
|
|
{
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsPath str;
|
|
|
|
v9fs_path_init(&str);
|
|
|
|
v9fs_path_copy(&str, dst);
|
|
|
|
v9fs_string_sprintf((V9fsString *)dst, "%s%s", src->data, str.data+len);
|
|
|
|
v9fs_path_free(&str);
|
|
|
|
/* +1 to include terminating NULL */
|
|
|
|
dst->size++;
|
2010-04-29 14:15:00 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_version(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:51 +02:00
|
|
|
V9fsString version;
|
|
|
|
size_t offset = 7;
|
|
|
|
|
2010-06-09 15:44:28 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "ds", &s->msize, &version);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_version(pdu->tag, pdu->id, s->msize, version.data);
|
2010-04-29 14:14:51 +02:00
|
|
|
|
2010-05-27 10:27:29 +02:00
|
|
|
if (!strcmp(version.data, "9P2000.u")) {
|
|
|
|
s->proto_version = V9FS_PROTO_2000U;
|
|
|
|
} else if (!strcmp(version.data, "9P2000.L")) {
|
|
|
|
s->proto_version = V9FS_PROTO_2000L;
|
|
|
|
} else {
|
2010-04-29 14:14:51 +02:00
|
|
|
v9fs_string_sprintf(&version, "unknown");
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
2010-04-29 14:14:51 +02:00
|
|
|
|
2010-06-09 15:44:28 +02:00
|
|
|
offset += pdu_marshal(pdu, offset, "ds", s->msize, &version);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_version_return(pdu->tag, pdu->id, s->msize, version.data);
|
|
|
|
|
2010-04-29 14:14:51 +02:00
|
|
|
complete_pdu(s, pdu, offset);
|
|
|
|
|
|
|
|
v9fs_string_free(&version);
|
2011-05-18 23:18:05 +02:00
|
|
|
return;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_attach(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:52 +02:00
|
|
|
int32_t fid, afid, n_uname;
|
|
|
|
V9fsString uname, aname;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
size_t offset = 7;
|
2011-05-08 09:45:29 +02:00
|
|
|
V9fsQID qid;
|
2010-04-29 14:14:52 +02:00
|
|
|
ssize_t err;
|
|
|
|
|
|
|
|
pdu_unmarshal(pdu, offset, "ddssd", &fid, &afid, &uname, &aname, &n_uname);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_attach(pdu->tag, pdu->id, fid, afid, uname.data, aname.data);
|
2010-04-29 14:14:52 +02:00
|
|
|
|
|
|
|
fidp = alloc_fid(s, fid);
|
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
2010-04-29 14:14:52 +02:00
|
|
|
fidp->uid = n_uname;
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, NULL, "/", &fidp->path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
err = -EINVAL;
|
|
|
|
clunk_fid(s, fid);
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = fid_to_qid(pdu, fidp, &qid);
|
2011-05-08 09:45:29 +02:00
|
|
|
if (err < 0) {
|
2010-04-29 14:14:52 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
clunk_fid(s, fid);
|
2010-04-29 14:14:52 +02:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
offset += pdu_marshal(pdu, offset, "Q", &qid);
|
|
|
|
err = offset;
|
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_attach_return(pdu->tag, pdu->id,
|
|
|
|
qid.type, qid.version, qid.path);
|
2010-04-29 14:14:52 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
v9fs_string_free(&uname);
|
|
|
|
v9fs_string_free(&aname);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_stat(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2010-04-29 14:14:53 +02:00
|
|
|
int32_t fid;
|
2011-05-07 14:59:24 +02:00
|
|
|
V9fsStat v9stat;
|
2010-04-29 14:14:53 +02:00
|
|
|
ssize_t err = 0;
|
2011-05-07 14:59:24 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
struct stat stbuf;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:53 +02:00
|
|
|
|
2011-05-07 14:59:24 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "d", &fid);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_stat(pdu->tag, pdu->id, fid);
|
2011-05-18 14:08:07 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-07 14:59:24 +02:00
|
|
|
if (fidp == NULL) {
|
2010-04-29 14:14:53 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lstat(pdu, &fidp->path, &stbuf);
|
2011-05-07 14:59:24 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = stat_to_v9stat(pdu, &fidp->path, &stbuf, &v9stat);
|
2011-05-07 14:59:24 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
offset += pdu_marshal(pdu, offset, "wS", 0, &v9stat);
|
|
|
|
err = offset;
|
|
|
|
v9fs_stat_free(&v9stat);
|
2010-04-29 14:14:53 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_stat_return(pdu->tag, pdu->id, v9stat.mode,
|
|
|
|
v9stat.atime, v9stat.mtime, v9stat.length);
|
|
|
|
|
2011-05-07 14:59:24 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_getattr(void *opaque)
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
{
|
|
|
|
int32_t fid;
|
2011-05-19 01:04:33 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
ssize_t retval = 0;
|
|
|
|
struct stat stbuf;
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
V9fsFidState *fidp;
|
|
|
|
uint64_t request_mask;
|
2011-05-19 01:04:33 +02:00
|
|
|
V9fsStatDotl v9stat_dotl;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
|
2011-05-19 01:04:33 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dq", &fid, &request_mask);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_getattr(pdu->tag, pdu->id, fid, request_mask);
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
if (fidp == NULL) {
|
2011-05-19 01:04:33 +02:00
|
|
|
retval = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
}
|
2011-05-19 01:04:33 +02:00
|
|
|
/*
|
|
|
|
* Currently we only support BASIC fields in stat, so there is no
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
* need to look at request_mask.
|
|
|
|
*/
|
2011-08-02 08:06:17 +02:00
|
|
|
retval = v9fs_co_lstat(pdu, &fidp->path, &stbuf);
|
2011-05-19 01:04:33 +02:00
|
|
|
if (retval < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
stat_to_v9stat_dotl(s, &stbuf, &v9stat_dotl);
|
2011-10-12 15:41:25 +02:00
|
|
|
|
|
|
|
/* fill st_gen if requested and supported by underlying fs */
|
|
|
|
if (request_mask & P9_STATS_GEN) {
|
|
|
|
retval = v9fs_co_st_gen(pdu, &fidp->path, stbuf.st_mode, &v9stat_dotl);
|
|
|
|
if (retval < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9stat_dotl.st_result_mask |= P9_STATS_GEN;
|
|
|
|
}
|
2011-05-19 01:04:33 +02:00
|
|
|
retval = offset;
|
|
|
|
retval += pdu_marshal(pdu, offset, "A", &v9stat_dotl);
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_getattr_return(pdu->tag, pdu->id, v9stat_dotl.st_result_mask,
|
|
|
|
v9stat_dotl.st_mode, v9stat_dotl.st_uid,
|
|
|
|
v9stat_dotl.st_gid);
|
|
|
|
|
2011-05-19 01:04:33 +02:00
|
|
|
complete_pdu(s, pdu, retval);
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
}
|
|
|
|
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
/* From Linux kernel code */
|
|
|
|
#define ATTR_MODE (1 << 0)
|
|
|
|
#define ATTR_UID (1 << 1)
|
|
|
|
#define ATTR_GID (1 << 2)
|
|
|
|
#define ATTR_SIZE (1 << 3)
|
|
|
|
#define ATTR_ATIME (1 << 4)
|
|
|
|
#define ATTR_MTIME (1 << 5)
|
|
|
|
#define ATTR_CTIME (1 << 6)
|
|
|
|
#define ATTR_MASK 127
|
|
|
|
#define ATTR_ATIME_SET (1 << 7)
|
|
|
|
#define ATTR_MTIME_SET (1 << 8)
|
|
|
|
|
2011-05-19 01:05:10 +02:00
|
|
|
static void v9fs_setattr(void *opaque)
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
{
|
2011-05-19 01:05:10 +02:00
|
|
|
int err = 0;
|
|
|
|
int32_t fid;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
size_t offset = 7;
|
|
|
|
V9fsIattr v9iattr;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
|
2011-05-19 01:05:10 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dI", &fid, &v9iattr);
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-19 01:05:10 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
}
|
2011-05-19 01:05:10 +02:00
|
|
|
if (v9iattr.valid & ATTR_MODE) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_chmod(pdu, &fidp->path, v9iattr.mode);
|
2011-05-19 01:05:10 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
}
|
|
|
|
}
|
2011-05-19 01:05:10 +02:00
|
|
|
if (v9iattr.valid & (ATTR_ATIME | ATTR_MTIME)) {
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
struct timespec times[2];
|
2011-05-19 01:05:10 +02:00
|
|
|
if (v9iattr.valid & ATTR_ATIME) {
|
|
|
|
if (v9iattr.valid & ATTR_ATIME_SET) {
|
|
|
|
times[0].tv_sec = v9iattr.atime_sec;
|
|
|
|
times[0].tv_nsec = v9iattr.atime_nsec;
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
} else {
|
|
|
|
times[0].tv_nsec = UTIME_NOW;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
times[0].tv_nsec = UTIME_OMIT;
|
|
|
|
}
|
2011-05-19 01:05:10 +02:00
|
|
|
if (v9iattr.valid & ATTR_MTIME) {
|
|
|
|
if (v9iattr.valid & ATTR_MTIME_SET) {
|
|
|
|
times[1].tv_sec = v9iattr.mtime_sec;
|
|
|
|
times[1].tv_nsec = v9iattr.mtime_nsec;
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
} else {
|
|
|
|
times[1].tv_nsec = UTIME_NOW;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
times[1].tv_nsec = UTIME_OMIT;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_utimensat(pdu, &fidp->path, times);
|
2011-05-19 01:05:10 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
}
|
2011-05-19 01:05:10 +02:00
|
|
|
/*
|
|
|
|
* If the only valid entry in iattr is ctime we can call
|
|
|
|
* chown(-1,-1) to update the ctime of the file
|
|
|
|
*/
|
|
|
|
if ((v9iattr.valid & (ATTR_UID | ATTR_GID)) ||
|
|
|
|
((v9iattr.valid & ATTR_CTIME)
|
|
|
|
&& !((v9iattr.valid & ATTR_MASK) & ~ATTR_CTIME))) {
|
|
|
|
if (!(v9iattr.valid & ATTR_UID)) {
|
|
|
|
v9iattr.uid = -1;
|
|
|
|
}
|
|
|
|
if (!(v9iattr.valid & ATTR_GID)) {
|
|
|
|
v9iattr.gid = -1;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_chown(pdu, &fidp->path, v9iattr.uid,
|
2011-05-19 01:05:10 +02:00
|
|
|
v9iattr.gid);
|
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
}
|
2011-05-19 01:05:10 +02:00
|
|
|
if (v9iattr.valid & (ATTR_SIZE)) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_truncate(pdu, &fidp->path, v9iattr.size);
|
2011-05-19 01:05:10 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
}
|
2011-05-19 01:05:10 +02:00
|
|
|
err = offset;
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-05-19 01:05:10 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
}
|
|
|
|
|
2011-05-07 16:11:55 +02:00
|
|
|
static int v9fs_walk_marshal(V9fsPDU *pdu, uint16_t nwnames, V9fsQID *qids)
|
2010-04-29 14:14:54 +02:00
|
|
|
{
|
|
|
|
int i;
|
2011-05-07 16:11:55 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
offset += pdu_marshal(pdu, offset, "w", nwnames);
|
|
|
|
for (i = 0; i < nwnames; i++) {
|
|
|
|
offset += pdu_marshal(pdu, offset, "Q", &qids[i]);
|
2010-04-29 14:14:54 +02:00
|
|
|
}
|
2011-05-07 16:11:55 +02:00
|
|
|
return offset;
|
2010-04-29 14:14:54 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_walk(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2011-05-07 16:11:55 +02:00
|
|
|
int name_idx;
|
|
|
|
V9fsQID *qids = NULL;
|
|
|
|
int i, err = 0;
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsPath dpath, path;
|
2011-05-07 16:11:55 +02:00
|
|
|
uint16_t nwnames;
|
|
|
|
struct stat stbuf;
|
|
|
|
size_t offset = 7;
|
|
|
|
int32_t fid, newfid;
|
|
|
|
V9fsString *wnames = NULL;
|
|
|
|
V9fsFidState *fidp;
|
2011-05-18 14:08:07 +02:00
|
|
|
V9fsFidState *newfidp = NULL;;
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:54 +02:00
|
|
|
|
2011-05-07 16:11:55 +02:00
|
|
|
offset += pdu_unmarshal(pdu, offset, "ddw", &fid,
|
|
|
|
&newfid, &nwnames);
|
2010-04-29 14:14:54 +02:00
|
|
|
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_walk(pdu->tag, pdu->id, fid, newfid, nwnames);
|
|
|
|
|
2011-05-07 16:11:55 +02:00
|
|
|
if (nwnames && nwnames <= P9_MAXWELEM) {
|
|
|
|
wnames = g_malloc0(sizeof(wnames[0]) * nwnames);
|
|
|
|
qids = g_malloc0(sizeof(qids[0]) * nwnames);
|
|
|
|
for (i = 0; i < nwnames; i++) {
|
|
|
|
offset += pdu_unmarshal(pdu, offset, "s", &wnames[i]);
|
2010-04-29 14:14:54 +02:00
|
|
|
}
|
2011-05-07 16:11:55 +02:00
|
|
|
} else if (nwnames > P9_MAXWELEM) {
|
2011-04-14 11:24:40 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:54 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-07 16:11:55 +02:00
|
|
|
if (fidp == NULL) {
|
2010-04-29 14:14:54 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:54 +02:00
|
|
|
}
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_init(&dpath);
|
|
|
|
v9fs_path_init(&path);
|
|
|
|
/*
|
|
|
|
* Both dpath and path initially poin to fidp.
|
|
|
|
* Needed to handle request with nwnames == 0
|
|
|
|
*/
|
|
|
|
v9fs_path_copy(&dpath, &fidp->path);
|
|
|
|
v9fs_path_copy(&path, &fidp->path);
|
|
|
|
for (name_idx = 0; name_idx < nwnames; name_idx++) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &dpath, wnames[name_idx].data, &path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lstat(pdu, &path, &stbuf);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
stat_to_qid(&stbuf, &qids[name_idx]);
|
|
|
|
v9fs_path_copy(&dpath, &path);
|
|
|
|
}
|
2010-04-29 14:14:54 +02:00
|
|
|
if (fid == newfid) {
|
2011-05-07 16:11:55 +02:00
|
|
|
BUG_ON(fidp->fid_type != P9_FID_NONE);
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_copy(&fidp->path, &path);
|
2010-04-29 14:14:54 +02:00
|
|
|
} else {
|
2011-05-07 16:11:55 +02:00
|
|
|
newfidp = alloc_fid(s, newfid);
|
|
|
|
if (newfidp == NULL) {
|
2010-04-29 14:14:54 +02:00
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-05-07 16:11:55 +02:00
|
|
|
newfidp->uid = fidp->uid;
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_copy(&newfidp->path, &path);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
2011-05-07 16:11:55 +02:00
|
|
|
err = v9fs_walk_marshal(pdu, nwnames, qids);
|
2010-04-29 14:14:54 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
if (newfidp) {
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, newfidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
}
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&dpath);
|
|
|
|
v9fs_path_free(&path);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_walk_return(pdu->tag, pdu->id, nwnames, qids);
|
2011-05-07 16:11:55 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
if (nwnames && nwnames <= P9_MAXWELEM) {
|
|
|
|
for (name_idx = 0; name_idx < nwnames; name_idx++) {
|
|
|
|
v9fs_string_free(&wnames[name_idx]);
|
|
|
|
}
|
|
|
|
g_free(wnames);
|
|
|
|
g_free(qids);
|
|
|
|
}
|
2011-09-09 11:44:18 +02:00
|
|
|
return;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int32_t get_iounit(V9fsPDU *pdu, V9fsPath *path)
|
2010-06-09 15:44:28 +02:00
|
|
|
{
|
|
|
|
struct statfs stbuf;
|
|
|
|
int32_t iounit = 0;
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsState *s = pdu->s;
|
2010-06-09 15:44:28 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* iounit should be multiples of f_bsize (host filesystem block size
|
|
|
|
* and as well as less than (client msize - P9_IOHDRSZ))
|
|
|
|
*/
|
2011-08-02 08:06:17 +02:00
|
|
|
if (!v9fs_co_statfs(pdu, path, &stbuf)) {
|
2010-06-09 15:44:28 +02:00
|
|
|
iounit = stbuf.f_bsize;
|
|
|
|
iounit *= (s->msize - P9_IOHDRSZ)/stbuf.f_bsize;
|
|
|
|
}
|
|
|
|
if (!iounit) {
|
|
|
|
iounit = s->msize - P9_IOHDRSZ;
|
|
|
|
}
|
|
|
|
return iounit;
|
|
|
|
}
|
|
|
|
|
2011-05-07 14:06:36 +02:00
|
|
|
static void v9fs_open(void *opaque)
|
2010-06-09 15:44:28 +02:00
|
|
|
{
|
2011-05-07 14:06:36 +02:00
|
|
|
int flags;
|
|
|
|
int iounit;
|
|
|
|
int32_t fid;
|
|
|
|
int32_t mode;
|
|
|
|
V9fsQID qid;
|
|
|
|
ssize_t err = 0;
|
|
|
|
size_t offset = 7;
|
|
|
|
struct stat stbuf;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-06-09 15:44:28 +02:00
|
|
|
|
2011-05-07 14:06:36 +02:00
|
|
|
if (s->proto_version == V9FS_PROTO_2000L) {
|
|
|
|
pdu_unmarshal(pdu, offset, "dd", &fid, &mode);
|
|
|
|
} else {
|
|
|
|
pdu_unmarshal(pdu, offset, "db", &fid, &mode);
|
|
|
|
}
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_open(pdu->tag, pdu->id, fid, mode);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-07 14:06:36 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:55 +02:00
|
|
|
}
|
2011-05-07 14:06:36 +02:00
|
|
|
BUG_ON(fidp->fid_type != P9_FID_NONE);
|
2010-04-29 14:14:55 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lstat(pdu, &fidp->path, &stbuf);
|
2011-05-07 14:06:36 +02:00
|
|
|
if (err < 0) {
|
2010-04-29 14:14:55 +02:00
|
|
|
goto out;
|
|
|
|
}
|
2011-05-07 14:06:36 +02:00
|
|
|
stat_to_qid(&stbuf, &qid);
|
|
|
|
if (S_ISDIR(stbuf.st_mode)) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_opendir(pdu, fidp);
|
2011-05-07 14:06:36 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
fidp->fid_type = P9_FID_DIR;
|
|
|
|
offset += pdu_marshal(pdu, offset, "Qd", &qid, 0);
|
|
|
|
err = offset;
|
2010-04-29 14:14:55 +02:00
|
|
|
} else {
|
2010-06-22 16:17:04 +02:00
|
|
|
if (s->proto_version == V9FS_PROTO_2000L) {
|
2011-10-12 15:41:23 +02:00
|
|
|
flags = get_dotl_openflags(s, mode);
|
2010-06-22 16:17:04 +02:00
|
|
|
} else {
|
2011-05-07 14:06:36 +02:00
|
|
|
flags = omode_to_uflags(mode);
|
2010-06-22 16:17:04 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_open(pdu, fidp, flags);
|
2011-05-07 14:06:36 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
fidp->fid_type = P9_FID_FILE;
|
2011-05-18 12:10:57 +02:00
|
|
|
fidp->open_flags = flags;
|
|
|
|
if (flags & O_EXCL) {
|
|
|
|
/*
|
|
|
|
* We let the host file system do O_EXCL check
|
|
|
|
* We should not reclaim such fd
|
|
|
|
*/
|
|
|
|
fidp->flags |= FID_NON_RECLAIMABLE;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
iounit = get_iounit(pdu, &fidp->path);
|
2011-05-07 14:06:36 +02:00
|
|
|
offset += pdu_marshal(pdu, offset, "Qd", &qid, iounit);
|
|
|
|
err = offset;
|
2010-04-29 14:14:55 +02:00
|
|
|
}
|
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_open_return(pdu->tag, pdu->id,
|
|
|
|
qid.type, qid.version, qid.path, iounit);
|
2010-04-29 14:14:55 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_lcreate(void *opaque)
|
2010-06-18 03:27:24 +02:00
|
|
|
{
|
|
|
|
int32_t dfid, flags, mode;
|
|
|
|
gid_t gid;
|
|
|
|
ssize_t err = 0;
|
2011-08-08 20:24:08 +02:00
|
|
|
ssize_t offset = 7;
|
|
|
|
V9fsString name;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
struct stat stbuf;
|
|
|
|
V9fsQID qid;
|
|
|
|
int32_t iounit;
|
|
|
|
V9fsPDU *pdu = opaque;
|
2010-06-18 03:27:24 +02:00
|
|
|
|
2011-08-08 20:24:08 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dsddd", &dfid, &name, &flags,
|
|
|
|
&mode, &gid);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_lcreate(pdu->tag, pdu->id, dfid, flags, mode, gid);
|
2010-06-18 03:27:24 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, dfid);
|
2011-08-08 20:24:08 +02:00
|
|
|
if (fidp == NULL) {
|
2010-06-18 03:27:24 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-06-18 03:27:24 +02:00
|
|
|
}
|
|
|
|
|
2011-10-12 15:41:23 +02:00
|
|
|
flags = get_dotl_openflags(pdu->s, flags);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_open2(pdu, fidp, &name, gid,
|
2011-05-24 11:40:56 +02:00
|
|
|
flags | O_CREAT, mode, &stbuf);
|
2011-08-08 20:24:08 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
fidp->fid_type = P9_FID_FILE;
|
2011-05-18 12:10:57 +02:00
|
|
|
fidp->open_flags = flags;
|
|
|
|
if (flags & O_EXCL) {
|
|
|
|
/*
|
|
|
|
* We let the host file system do O_EXCL check
|
|
|
|
* We should not reclaim such fd
|
|
|
|
*/
|
|
|
|
fidp->flags |= FID_NON_RECLAIMABLE;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
iounit = get_iounit(pdu, &fidp->path);
|
2011-08-08 20:24:08 +02:00
|
|
|
stat_to_qid(&stbuf, &qid);
|
|
|
|
offset += pdu_marshal(pdu, offset, "Qd", &qid, iounit);
|
|
|
|
err = offset;
|
2010-06-18 03:27:24 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_lcreate_return(pdu->tag, pdu->id,
|
|
|
|
qid.type, qid.version, qid.path, iounit);
|
2011-08-08 20:24:08 +02:00
|
|
|
complete_pdu(pdu->s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
2010-06-18 03:27:24 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_fsync(void *opaque)
|
2010-09-23 02:18:33 +02:00
|
|
|
{
|
2011-05-07 17:55:51 +02:00
|
|
|
int err;
|
2010-09-23 02:18:33 +02:00
|
|
|
int32_t fid;
|
2011-05-07 17:55:51 +02:00
|
|
|
int datasync;
|
2010-09-23 02:18:33 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
V9fsFidState *fidp;
|
2011-05-07 17:55:51 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-09-23 02:18:33 +02:00
|
|
|
|
2010-10-22 19:08:45 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dd", &fid, &datasync);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_fsync(pdu->tag, pdu->id, fid, datasync);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2010-09-23 02:18:33 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-09-23 02:18:33 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_fsync(pdu, fidp, datasync);
|
2011-05-07 17:55:51 +02:00
|
|
|
if (!err) {
|
|
|
|
err = offset;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-05-07 17:55:51 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
2010-09-23 02:18:33 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_clunk(void *opaque)
|
2010-04-29 14:14:55 +02:00
|
|
|
{
|
2011-05-07 17:06:38 +02:00
|
|
|
int err;
|
2010-04-29 14:14:57 +02:00
|
|
|
int32_t fid;
|
|
|
|
size_t offset = 7;
|
2011-05-18 14:08:07 +02:00
|
|
|
V9fsFidState *fidp;
|
2011-05-07 17:06:38 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:57 +02:00
|
|
|
|
|
|
|
pdu_unmarshal(pdu, offset, "d", &fid);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_clunk(pdu->tag, pdu->id, fid);
|
2011-05-18 14:08:07 +02:00
|
|
|
|
2011-08-02 08:06:24 +02:00
|
|
|
fidp = clunk_fid(s, fid);
|
2011-05-18 14:08:07 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -ENOENT;
|
|
|
|
goto out_nofid;
|
|
|
|
}
|
2011-08-02 08:06:24 +02:00
|
|
|
/*
|
|
|
|
* Bump the ref so that put_fid will
|
|
|
|
* free the fid.
|
|
|
|
*/
|
|
|
|
fidp->ref++;
|
2010-04-29 14:14:57 +02:00
|
|
|
err = offset;
|
2011-08-02 08:06:24 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2010-04-29 14:14:57 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-20 22:46:31 +02:00
|
|
|
static int v9fs_xattr_read(V9fsState *s, V9fsPDU *pdu,
|
|
|
|
V9fsFidState *fidp, int64_t off, int32_t max_count)
|
2010-04-29 14:14:56 +02:00
|
|
|
{
|
2011-05-20 22:46:31 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
int read_count;
|
|
|
|
int64_t xattr_len;
|
2010-04-29 14:14:56 +02:00
|
|
|
|
2011-05-20 22:46:31 +02:00
|
|
|
xattr_len = fidp->fs.xattr.len;
|
|
|
|
read_count = xattr_len - off;
|
|
|
|
if (read_count > max_count) {
|
|
|
|
read_count = max_count;
|
|
|
|
} else if (read_count < 0) {
|
|
|
|
/*
|
|
|
|
* read beyond XATTR value
|
|
|
|
*/
|
|
|
|
read_count = 0;
|
2010-04-29 14:14:56 +02:00
|
|
|
}
|
2011-05-20 22:46:31 +02:00
|
|
|
offset += pdu_marshal(pdu, offset, "d", read_count);
|
|
|
|
offset += pdu_pack(pdu, offset,
|
|
|
|
((char *)fidp->fs.xattr.value) + off,
|
|
|
|
read_count);
|
|
|
|
return offset;
|
2010-04-29 14:14:56 +02:00
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int v9fs_do_readdir_with_stat(V9fsPDU *pdu,
|
2011-05-20 22:46:31 +02:00
|
|
|
V9fsFidState *fidp, int32_t max_count)
|
2010-04-29 14:14:56 +02:00
|
|
|
{
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsPath path;
|
2011-05-20 22:46:31 +02:00
|
|
|
V9fsStat v9stat;
|
|
|
|
int len, err = 0;
|
|
|
|
int32_t count = 0;
|
|
|
|
struct stat stbuf;
|
|
|
|
off_t saved_dir_pos;
|
2011-05-18 13:53:00 +02:00
|
|
|
struct dirent *dent, *result;
|
2010-04-29 14:14:56 +02:00
|
|
|
|
2011-05-20 22:46:31 +02:00
|
|
|
/* save the directory position */
|
2011-08-02 08:06:17 +02:00
|
|
|
saved_dir_pos = v9fs_co_telldir(pdu, fidp);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (saved_dir_pos < 0) {
|
|
|
|
return saved_dir_pos;
|
2010-04-29 14:14:56 +02:00
|
|
|
}
|
2011-05-18 13:53:00 +02:00
|
|
|
|
|
|
|
dent = g_malloc(sizeof(struct dirent));
|
|
|
|
|
2011-05-20 22:46:31 +02:00
|
|
|
while (1) {
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_init(&path);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_readdir_r(pdu, fidp, dent, &result);
|
2011-05-18 13:53:00 +02:00
|
|
|
if (err || !result) {
|
2011-05-20 22:46:31 +02:00
|
|
|
break;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &fidp->path, dent->d_name, &path);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lstat(pdu, &path, &stbuf);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = stat_to_v9stat(pdu, &path, &stbuf, &v9stat);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
2010-04-29 14:14:56 +02:00
|
|
|
}
|
2011-05-20 22:46:31 +02:00
|
|
|
/* 11 = 7 + 4 (7 = start offset, 4 = space for storing count) */
|
|
|
|
len = pdu_marshal(pdu, 11 + count, "S", &v9stat);
|
|
|
|
if ((len != (v9stat.size + 2)) || ((count + len) > max_count)) {
|
|
|
|
/* Ran out of buffer. Set dir back to old position and return */
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_seekdir(pdu, fidp, saved_dir_pos);
|
2011-05-20 22:46:31 +02:00
|
|
|
v9fs_stat_free(&v9stat);
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&path);
|
2011-05-18 13:53:00 +02:00
|
|
|
g_free(dent);
|
2011-05-20 22:46:31 +02:00
|
|
|
return count;
|
|
|
|
}
|
|
|
|
count += len;
|
|
|
|
v9fs_stat_free(&v9stat);
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&path);
|
2011-05-20 22:46:31 +02:00
|
|
|
saved_dir_pos = dent->d_off;
|
2010-04-29 14:14:56 +02:00
|
|
|
}
|
|
|
|
out:
|
2011-05-18 13:53:00 +02:00
|
|
|
g_free(dent);
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&path);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return err;
|
2010-09-02 07:39:06 +02:00
|
|
|
}
|
2011-05-20 22:46:31 +02:00
|
|
|
return count;
|
2010-09-02 07:39:06 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_read(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2010-04-29 14:14:56 +02:00
|
|
|
int32_t fid;
|
2011-05-20 22:46:31 +02:00
|
|
|
int64_t off;
|
2010-04-29 14:14:56 +02:00
|
|
|
ssize_t err = 0;
|
2011-05-20 22:46:31 +02:00
|
|
|
int32_t count = 0;
|
|
|
|
size_t offset = 7;
|
|
|
|
int32_t max_count;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:56 +02:00
|
|
|
|
2011-05-20 22:46:31 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dqd", &fid, &off, &max_count);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_read(pdu->tag, pdu->id, fid, off, max_count);
|
2011-05-18 14:08:07 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (fidp == NULL) {
|
2010-04-29 14:14:56 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:56 +02:00
|
|
|
}
|
2011-05-20 22:46:31 +02:00
|
|
|
if (fidp->fid_type == P9_FID_DIR) {
|
2010-04-29 14:14:56 +02:00
|
|
|
|
2011-05-20 22:46:31 +02:00
|
|
|
if (off == 0) {
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_rewinddir(pdu, fidp);
|
2010-04-29 14:14:56 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
count = v9fs_do_readdir_with_stat(pdu, fidp, max_count);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (count < 0) {
|
|
|
|
err = count;
|
|
|
|
goto out;
|
2010-10-08 08:00:16 +02:00
|
|
|
}
|
2011-05-20 22:46:31 +02:00
|
|
|
err = offset;
|
|
|
|
err += pdu_marshal(pdu, offset, "d", count);
|
|
|
|
err += count;
|
|
|
|
} else if (fidp->fid_type == P9_FID_FILE) {
|
|
|
|
int32_t cnt;
|
|
|
|
int32_t len;
|
|
|
|
struct iovec *sg;
|
|
|
|
struct iovec iov[128]; /* FIXME: bad, bad, bad */
|
|
|
|
|
|
|
|
sg = iov;
|
|
|
|
pdu_marshal(pdu, offset + 4, "v", sg, &cnt);
|
|
|
|
sg = cap_sg(sg, max_count, &cnt);
|
|
|
|
do {
|
|
|
|
if (0) {
|
|
|
|
print_sg(sg, cnt);
|
|
|
|
}
|
|
|
|
/* Loop in case of EINTR */
|
|
|
|
do {
|
2011-08-02 08:06:17 +02:00
|
|
|
len = v9fs_co_preadv(pdu, fidp, sg, cnt, off);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (len >= 0) {
|
|
|
|
off += len;
|
|
|
|
count += len;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
} while (len == -EINTR && !pdu->cancelled);
|
2011-05-20 22:46:31 +02:00
|
|
|
if (len < 0) {
|
|
|
|
/* IO error return the error */
|
|
|
|
err = len;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
sg = adjust_sg(sg, len, &cnt);
|
|
|
|
} while (count < max_count && len > 0);
|
|
|
|
err = offset;
|
|
|
|
err += pdu_marshal(pdu, offset, "d", count);
|
|
|
|
err += count;
|
|
|
|
} else if (fidp->fid_type == P9_FID_XATTR) {
|
|
|
|
err = v9fs_xattr_read(s, pdu, fidp, off, max_count);
|
2010-04-29 14:14:56 +02:00
|
|
|
} else {
|
|
|
|
err = -EINVAL;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
2010-04-29 14:14:56 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_read_return(pdu->tag, pdu->id, count, err);
|
2010-04-29 14:14:56 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-19 00:57:17 +02:00
|
|
|
static size_t v9fs_readdir_data_size(V9fsString *name)
|
2010-06-09 11:27:57 +02:00
|
|
|
{
|
2011-05-19 00:57:17 +02:00
|
|
|
/*
|
|
|
|
* Size of each dirent on the wire: size of qid (13) + size of offset (8)
|
|
|
|
* size of type (1) + size of name.size (2) + strlen(name.data)
|
|
|
|
*/
|
|
|
|
return 24 + v9fs_string_size(name);
|
2010-06-09 11:27:57 +02:00
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int v9fs_do_readdir(V9fsPDU *pdu,
|
2011-05-19 00:57:17 +02:00
|
|
|
V9fsFidState *fidp, int32_t max_count)
|
2010-06-09 11:27:57 +02:00
|
|
|
{
|
|
|
|
size_t size;
|
2011-05-19 00:57:17 +02:00
|
|
|
V9fsQID qid;
|
|
|
|
V9fsString name;
|
|
|
|
int len, err = 0;
|
|
|
|
int32_t count = 0;
|
|
|
|
off_t saved_dir_pos;
|
2011-05-18 13:53:00 +02:00
|
|
|
struct dirent *dent, *result;
|
2010-06-09 11:27:57 +02:00
|
|
|
|
2011-05-19 00:57:17 +02:00
|
|
|
/* save the directory position */
|
2011-08-02 08:06:17 +02:00
|
|
|
saved_dir_pos = v9fs_co_telldir(pdu, fidp);
|
2011-05-19 00:57:17 +02:00
|
|
|
if (saved_dir_pos < 0) {
|
|
|
|
return saved_dir_pos;
|
|
|
|
}
|
2011-05-18 13:53:00 +02:00
|
|
|
|
|
|
|
dent = g_malloc(sizeof(struct dirent));
|
|
|
|
|
2011-05-19 00:57:17 +02:00
|
|
|
while (1) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_readdir_r(pdu, fidp, dent, &result);
|
2011-05-18 13:53:00 +02:00
|
|
|
if (err || !result) {
|
2011-05-19 00:57:17 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
v9fs_string_init(&name);
|
|
|
|
v9fs_string_sprintf(&name, "%s", dent->d_name);
|
|
|
|
if ((count + v9fs_readdir_data_size(&name)) > max_count) {
|
2010-06-09 11:27:57 +02:00
|
|
|
/* Ran out of buffer. Set dir back to old position and return */
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_seekdir(pdu, fidp, saved_dir_pos);
|
2011-05-19 00:57:17 +02:00
|
|
|
v9fs_string_free(&name);
|
2011-05-18 13:53:00 +02:00
|
|
|
g_free(dent);
|
2011-05-19 00:57:17 +02:00
|
|
|
return count;
|
2010-06-09 11:27:57 +02:00
|
|
|
}
|
2011-05-19 00:57:17 +02:00
|
|
|
/*
|
|
|
|
* Fill up just the path field of qid because the client uses
|
2010-06-09 11:27:57 +02:00
|
|
|
* only that. To fill the entire qid structure we will have
|
|
|
|
* to stat each dirent found, which is expensive
|
|
|
|
*/
|
2011-05-19 00:57:17 +02:00
|
|
|
size = MIN(sizeof(dent->d_ino), sizeof(qid.path));
|
|
|
|
memcpy(&qid.path, &dent->d_ino, size);
|
2010-06-09 11:27:57 +02:00
|
|
|
/* Fill the other fields with dummy values */
|
2011-05-19 00:57:17 +02:00
|
|
|
qid.type = 0;
|
|
|
|
qid.version = 0;
|
2010-06-09 11:27:57 +02:00
|
|
|
|
2011-05-19 00:57:17 +02:00
|
|
|
/* 11 = 7 + 4 (7 = start offset, 4 = space for storing count) */
|
|
|
|
len = pdu_marshal(pdu, 11 + count, "Qqbs",
|
|
|
|
&qid, dent->d_off,
|
|
|
|
dent->d_type, &name);
|
|
|
|
count += len;
|
|
|
|
v9fs_string_free(&name);
|
|
|
|
saved_dir_pos = dent->d_off;
|
|
|
|
}
|
2011-05-18 13:53:00 +02:00
|
|
|
g_free(dent);
|
2011-05-19 00:57:17 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
return count;
|
2010-06-09 11:27:57 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_readdir(void *opaque)
|
2010-06-09 11:27:57 +02:00
|
|
|
{
|
|
|
|
int32_t fid;
|
2011-05-19 00:57:17 +02:00
|
|
|
V9fsFidState *fidp;
|
|
|
|
ssize_t retval = 0;
|
2010-06-09 11:27:57 +02:00
|
|
|
size_t offset = 7;
|
2011-05-19 00:57:17 +02:00
|
|
|
int64_t initial_offset;
|
|
|
|
int32_t count, max_count;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-06-09 11:27:57 +02:00
|
|
|
|
2011-05-19 00:57:17 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dqd", &fid, &initial_offset, &max_count);
|
2010-06-09 11:27:57 +02:00
|
|
|
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_readdir(pdu->tag, pdu->id, fid, initial_offset, max_count);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-18 14:08:07 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
retval = -EINVAL;
|
|
|
|
goto out_nofid;
|
|
|
|
}
|
|
|
|
if (!fidp->fs.dir) {
|
2011-05-19 00:57:17 +02:00
|
|
|
retval = -EINVAL;
|
2010-06-09 11:27:57 +02:00
|
|
|
goto out;
|
|
|
|
}
|
2011-05-19 00:57:17 +02:00
|
|
|
if (initial_offset == 0) {
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_rewinddir(pdu, fidp);
|
2010-06-09 11:27:57 +02:00
|
|
|
} else {
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_seekdir(pdu, fidp, initial_offset);
|
2010-06-09 11:27:57 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
count = v9fs_do_readdir(pdu, fidp, max_count);
|
2011-05-19 00:57:17 +02:00
|
|
|
if (count < 0) {
|
|
|
|
retval = count;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
retval = offset;
|
|
|
|
retval += pdu_marshal(pdu, offset, "d", count);
|
|
|
|
retval += count;
|
2010-06-09 11:27:57 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_readdir_return(pdu->tag, pdu->id, count, retval);
|
2011-05-19 00:57:17 +02:00
|
|
|
complete_pdu(s, pdu, retval);
|
2010-06-09 11:27:57 +02:00
|
|
|
}
|
|
|
|
|
2011-05-08 08:59:31 +02:00
|
|
|
static int v9fs_xattr_write(V9fsState *s, V9fsPDU *pdu, V9fsFidState *fidp,
|
|
|
|
int64_t off, int32_t count,
|
|
|
|
struct iovec *sg, int cnt)
|
2010-09-02 07:39:07 +02:00
|
|
|
{
|
|
|
|
int i, to_copy;
|
|
|
|
ssize_t err = 0;
|
|
|
|
int write_count;
|
|
|
|
int64_t xattr_len;
|
2011-05-08 08:59:31 +02:00
|
|
|
size_t offset = 7;
|
2010-09-02 07:39:07 +02:00
|
|
|
|
2011-05-08 08:59:31 +02:00
|
|
|
|
|
|
|
xattr_len = fidp->fs.xattr.len;
|
|
|
|
write_count = xattr_len - off;
|
|
|
|
if (write_count > count) {
|
|
|
|
write_count = count;
|
2010-09-02 07:39:07 +02:00
|
|
|
} else if (write_count < 0) {
|
|
|
|
/*
|
|
|
|
* write beyond XATTR value len specified in
|
|
|
|
* xattrcreate
|
|
|
|
*/
|
|
|
|
err = -ENOSPC;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-05-08 08:59:31 +02:00
|
|
|
offset += pdu_marshal(pdu, offset, "d", write_count);
|
|
|
|
err = offset;
|
|
|
|
fidp->fs.xattr.copied_len += write_count;
|
2010-09-02 07:39:07 +02:00
|
|
|
/*
|
|
|
|
* Now copy the content from sg list
|
|
|
|
*/
|
2011-05-08 08:59:31 +02:00
|
|
|
for (i = 0; i < cnt; i++) {
|
|
|
|
if (write_count > sg[i].iov_len) {
|
|
|
|
to_copy = sg[i].iov_len;
|
2010-09-02 07:39:07 +02:00
|
|
|
} else {
|
|
|
|
to_copy = write_count;
|
|
|
|
}
|
2011-05-08 08:59:31 +02:00
|
|
|
memcpy((char *)fidp->fs.xattr.value + off, sg[i].iov_base, to_copy);
|
2010-09-02 07:39:07 +02:00
|
|
|
/* updating vs->off since we are not using below */
|
2011-05-08 08:59:31 +02:00
|
|
|
off += to_copy;
|
2010-09-02 07:39:07 +02:00
|
|
|
write_count -= to_copy;
|
|
|
|
}
|
|
|
|
out:
|
2011-05-08 08:59:31 +02:00
|
|
|
return err;
|
2010-09-02 07:39:07 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_write(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2011-05-08 08:59:31 +02:00
|
|
|
int cnt;
|
|
|
|
ssize_t err;
|
|
|
|
int32_t fid;
|
|
|
|
int64_t off;
|
|
|
|
int32_t count;
|
|
|
|
int32_t len = 0;
|
|
|
|
int32_t total = 0;
|
|
|
|
size_t offset = 7;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
struct iovec iov[128]; /* FIXME: bad, bad, bad */
|
|
|
|
struct iovec *sg = iov;
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:14:58 +02:00
|
|
|
|
2011-05-08 08:59:31 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dqdv", &fid, &off, &count, sg, &cnt);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_write(pdu->tag, pdu->id, fid, off, count, cnt);
|
2011-05-18 14:08:07 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-08 08:59:31 +02:00
|
|
|
if (fidp == NULL) {
|
2010-04-29 14:14:58 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
2011-05-08 08:59:31 +02:00
|
|
|
if (fidp->fid_type == P9_FID_FILE) {
|
|
|
|
if (fidp->fs.fd == -1) {
|
2010-09-02 07:39:07 +02:00
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-05-08 08:59:31 +02:00
|
|
|
} else if (fidp->fid_type == P9_FID_XATTR) {
|
2010-09-02 07:39:07 +02:00
|
|
|
/*
|
|
|
|
* setxattr operation
|
|
|
|
*/
|
2011-05-08 08:59:31 +02:00
|
|
|
err = v9fs_xattr_write(s, pdu, fidp, off, count, sg, cnt);
|
|
|
|
goto out;
|
2010-09-02 07:39:07 +02:00
|
|
|
} else {
|
2010-04-29 14:14:58 +02:00
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-05-08 08:59:31 +02:00
|
|
|
sg = cap_sg(sg, count, &cnt);
|
|
|
|
do {
|
|
|
|
if (0) {
|
|
|
|
print_sg(sg, cnt);
|
2010-10-08 08:00:16 +02:00
|
|
|
}
|
2011-05-08 08:59:31 +02:00
|
|
|
/* Loop in case of EINTR */
|
|
|
|
do {
|
2011-08-02 08:06:17 +02:00
|
|
|
len = v9fs_co_pwritev(pdu, fidp, sg, cnt, off);
|
2011-05-08 08:59:31 +02:00
|
|
|
if (len >= 0) {
|
|
|
|
off += len;
|
|
|
|
total += len;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
} while (len == -EINTR && !pdu->cancelled);
|
2011-05-08 08:59:31 +02:00
|
|
|
if (len < 0) {
|
|
|
|
/* IO error return the error */
|
|
|
|
err = len;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
sg = adjust_sg(sg, len, &cnt);
|
|
|
|
} while (total < count && len > 0);
|
|
|
|
offset += pdu_marshal(pdu, offset, "d", total);
|
|
|
|
err = offset;
|
2010-04-29 14:14:58 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_write_return(pdu->tag, pdu->id, total, err);
|
2011-05-08 08:59:31 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-08-08 20:26:50 +02:00
|
|
|
static void v9fs_create(void *opaque)
|
2010-06-09 15:44:28 +02:00
|
|
|
{
|
2011-08-08 20:26:50 +02:00
|
|
|
int32_t fid;
|
|
|
|
int err = 0;
|
|
|
|
size_t offset = 7;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsQID qid;
|
|
|
|
int32_t perm;
|
|
|
|
int8_t mode;
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsPath path;
|
2011-08-08 20:26:50 +02:00
|
|
|
struct stat stbuf;
|
|
|
|
V9fsString name;
|
|
|
|
V9fsString extension;
|
|
|
|
int iounit;
|
|
|
|
V9fsPDU *pdu = opaque;
|
2010-04-29 14:14:59 +02:00
|
|
|
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_init(&path);
|
2010-04-29 14:14:59 +02:00
|
|
|
|
2011-08-08 20:26:50 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dsdbs", &fid, &name,
|
|
|
|
&perm, &mode, &extension);
|
2010-04-29 14:14:59 +02:00
|
|
|
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_create(pdu->tag, pdu->id, fid, name.data, perm, mode);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:59 +02:00
|
|
|
}
|
2011-08-08 20:26:50 +02:00
|
|
|
if (perm & P9_STAT_MODE_DIR) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_mkdir(pdu, fidp, &name, perm & 0777,
|
2011-05-24 11:40:56 +02:00
|
|
|
fidp->uid, -1, &stbuf);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &fidp->path, name.data, &path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9fs_path_copy(&fidp->path, &path);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_opendir(pdu, fidp);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
fidp->fid_type = P9_FID_DIR;
|
|
|
|
} else if (perm & P9_STAT_MODE_SYMLINK) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_symlink(pdu, fidp, &name,
|
2011-05-24 11:40:56 +02:00
|
|
|
extension.data, -1 , &stbuf);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &fidp->path, name.data, &path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9fs_path_copy(&fidp->path, &path);
|
2011-08-08 20:26:50 +02:00
|
|
|
} else if (perm & P9_STAT_MODE_LINK) {
|
2011-09-09 11:44:18 +02:00
|
|
|
int32_t ofid = atoi(extension.data);
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsFidState *ofidp = get_fid(pdu, ofid);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (ofidp == NULL) {
|
2011-08-08 20:26:50 +02:00
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_link(pdu, ofidp, fidp, &name);
|
|
|
|
put_fid(pdu, ofidp);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &fidp->path, name.data, &path);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
2011-09-09 11:44:18 +02:00
|
|
|
fidp->fid_type = P9_FID_NONE;
|
2011-08-08 20:26:50 +02:00
|
|
|
goto out;
|
2010-04-29 14:14:59 +02:00
|
|
|
}
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_copy(&fidp->path, &path);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lstat(pdu, &fidp->path, &stbuf);
|
2011-05-24 11:40:56 +02:00
|
|
|
if (err < 0) {
|
|
|
|
fidp->fid_type = P9_FID_NONE;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-08 20:26:50 +02:00
|
|
|
} else if (perm & P9_STAT_MODE_DEVICE) {
|
2010-04-29 14:14:59 +02:00
|
|
|
char ctype;
|
|
|
|
uint32_t major, minor;
|
|
|
|
mode_t nmode = 0;
|
|
|
|
|
2011-08-08 20:26:50 +02:00
|
|
|
if (sscanf(extension.data, "%c %u %u", &ctype, &major, &minor) != 3) {
|
2010-04-29 14:14:59 +02:00
|
|
|
err = -errno;
|
2011-08-08 20:26:50 +02:00
|
|
|
goto out;
|
2010-04-29 14:14:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (ctype) {
|
|
|
|
case 'c':
|
|
|
|
nmode = S_IFCHR;
|
|
|
|
break;
|
|
|
|
case 'b':
|
|
|
|
nmode = S_IFBLK;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
err = -EIO;
|
2011-08-08 20:26:50 +02:00
|
|
|
goto out;
|
|
|
|
}
|
2010-06-18 03:27:24 +02:00
|
|
|
|
2011-08-08 20:26:50 +02:00
|
|
|
nmode |= perm & 0777;
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_mknod(pdu, fidp, &name, fidp->uid, -1,
|
2011-05-24 11:40:56 +02:00
|
|
|
makedev(major, minor), nmode, &stbuf);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &fidp->path, name.data, &path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9fs_path_copy(&fidp->path, &path);
|
2011-08-08 20:26:50 +02:00
|
|
|
} else if (perm & P9_STAT_MODE_NAMED_PIPE) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_mknod(pdu, fidp, &name, fidp->uid, -1,
|
2011-05-24 11:40:56 +02:00
|
|
|
0, S_IFIFO | (perm & 0777), &stbuf);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &fidp->path, name.data, &path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9fs_path_copy(&fidp->path, &path);
|
2011-08-08 20:26:50 +02:00
|
|
|
} else if (perm & P9_STAT_MODE_SOCKET) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_mknod(pdu, fidp, &name, fidp->uid, -1,
|
2011-05-24 11:40:56 +02:00
|
|
|
0, S_IFSOCK | (perm & 0777), &stbuf);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &fidp->path, name.data, &path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9fs_path_copy(&fidp->path, &path);
|
2011-08-08 20:26:50 +02:00
|
|
|
} else {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_open2(pdu, fidp, &name, -1,
|
2011-05-24 11:40:56 +02:00
|
|
|
omode_to_uflags(mode)|O_CREAT, perm, &stbuf);
|
2011-08-08 20:26:50 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
fidp->fid_type = P9_FID_FILE;
|
2011-05-18 12:10:57 +02:00
|
|
|
fidp->open_flags = omode_to_uflags(mode);
|
|
|
|
if (fidp->open_flags & O_EXCL) {
|
|
|
|
/*
|
|
|
|
* We let the host file system do O_EXCL check
|
|
|
|
* We should not reclaim such fd
|
|
|
|
*/
|
|
|
|
fidp->flags |= FID_NON_RECLAIMABLE;
|
|
|
|
}
|
2010-04-29 14:14:59 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
iounit = get_iounit(pdu, &fidp->path);
|
2011-08-08 20:26:50 +02:00
|
|
|
stat_to_qid(&stbuf, &qid);
|
|
|
|
offset += pdu_marshal(pdu, offset, "Qd", &qid, iounit);
|
|
|
|
err = offset;
|
2010-04-29 14:14:59 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_create_return(pdu->tag, pdu->id,
|
|
|
|
qid.type, qid.version, qid.path, iounit);
|
2011-08-08 20:26:50 +02:00
|
|
|
complete_pdu(pdu->s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
|
|
|
v9fs_string_free(&extension);
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&path);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_symlink(void *opaque)
|
2010-06-09 23:02:08 +02:00
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
2011-08-08 20:30:01 +02:00
|
|
|
V9fsString name;
|
|
|
|
V9fsString symname;
|
|
|
|
V9fsFidState *dfidp;
|
|
|
|
V9fsQID qid;
|
|
|
|
struct stat stbuf;
|
2010-06-09 23:02:08 +02:00
|
|
|
int32_t dfid;
|
|
|
|
int err = 0;
|
|
|
|
gid_t gid;
|
2011-08-08 20:30:01 +02:00
|
|
|
size_t offset = 7;
|
2010-06-09 23:02:08 +02:00
|
|
|
|
2011-08-08 20:30:01 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dssd", &dfid, &name, &symname, &gid);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_symlink(pdu->tag, pdu->id, dfid, name.data, symname.data, gid);
|
2010-06-09 23:02:08 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
dfidp = get_fid(pdu, dfid);
|
2011-08-08 20:30:01 +02:00
|
|
|
if (dfidp == NULL) {
|
2010-06-09 23:02:08 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-06-09 23:02:08 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_symlink(pdu, dfidp, &name, symname.data, gid, &stbuf);
|
2011-08-08 20:30:01 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
stat_to_qid(&stbuf, &qid);
|
|
|
|
offset += pdu_marshal(pdu, offset, "Q", &qid);
|
|
|
|
err = offset;
|
2010-06-09 23:02:08 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, dfidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_symlink_return(pdu->tag, pdu->id,
|
|
|
|
qid.type, qid.version, qid.path);
|
2011-08-08 20:30:01 +02:00
|
|
|
complete_pdu(pdu->s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
|
|
|
v9fs_string_free(&symname);
|
2010-06-09 23:02:08 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_flush(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2011-08-02 08:06:17 +02:00
|
|
|
int16_t tag;
|
|
|
|
size_t offset = 7;
|
|
|
|
V9fsPDU *cancel_pdu;
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2011-08-02 08:06:17 +02:00
|
|
|
|
|
|
|
pdu_unmarshal(pdu, offset, "w", &tag);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_flush(pdu->tag, pdu->id, tag);
|
2011-08-02 08:06:17 +02:00
|
|
|
|
|
|
|
QLIST_FOREACH(cancel_pdu, &s->active_list, next) {
|
|
|
|
if (cancel_pdu->tag == tag) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cancel_pdu) {
|
|
|
|
cancel_pdu->cancelled = 1;
|
|
|
|
/*
|
|
|
|
* Wait for pdu to complete.
|
|
|
|
*/
|
|
|
|
qemu_co_queue_wait(&cancel_pdu->complete);
|
|
|
|
cancel_pdu->cancelled = 0;
|
|
|
|
free_pdu(pdu->s, cancel_pdu);
|
|
|
|
}
|
2010-04-29 14:15:02 +02:00
|
|
|
complete_pdu(s, pdu, 7);
|
2011-05-18 23:18:05 +02:00
|
|
|
return;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_link(void *opaque)
|
2010-06-09 20:21:15 +02:00
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-06-09 20:21:15 +02:00
|
|
|
int32_t dfid, oldfid;
|
|
|
|
V9fsFidState *dfidp, *oldfidp;
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsString name;;
|
2010-06-09 20:21:15 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
pdu_unmarshal(pdu, offset, "dds", &dfid, &oldfid, &name);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_link(pdu->tag, pdu->id, dfid, oldfid, name.data);
|
2010-06-09 20:21:15 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
dfidp = get_fid(pdu, dfid);
|
2010-06-09 20:21:15 +02:00
|
|
|
if (dfidp == NULL) {
|
2011-05-09 20:47:28 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-06-09 20:21:15 +02:00
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
oldfidp = get_fid(pdu, oldfid);
|
2010-06-09 20:21:15 +02:00
|
|
|
if (oldfidp == NULL) {
|
2011-05-09 20:47:28 +02:00
|
|
|
err = -ENOENT;
|
2010-06-09 20:21:15 +02:00
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_link(pdu, oldfidp, dfidp, &name);
|
2011-05-09 20:47:28 +02:00
|
|
|
if (!err) {
|
|
|
|
err = offset;
|
2010-06-09 20:21:15 +02:00
|
|
|
}
|
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, dfidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2010-06-09 20:21:15 +02:00
|
|
|
v9fs_string_free(&name);
|
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:05:54 +02:00
|
|
|
/* Only works with path name based fid */
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_remove(void *opaque)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
2010-04-29 14:15:01 +02:00
|
|
|
int32_t fid;
|
|
|
|
int err = 0;
|
2011-08-08 20:20:20 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
2010-04-29 14:15:01 +02:00
|
|
|
|
2011-08-08 20:20:20 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "d", &fid);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_remove(pdu->tag, pdu->id, fid);
|
2010-04-29 14:15:01 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-08-08 20:20:20 +02:00
|
|
|
if (fidp == NULL) {
|
2010-04-29 14:15:01 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
2011-08-02 08:05:54 +02:00
|
|
|
/* if fs driver is not path based, return EOPNOTSUPP */
|
2011-10-12 17:29:18 +02:00
|
|
|
if (!(pdu->s->ctx.export_flags & V9FS_PATHNAME_FSCONTEXT)) {
|
2011-08-02 08:05:54 +02:00
|
|
|
err = -EOPNOTSUPP;
|
|
|
|
goto out_err;
|
|
|
|
}
|
2011-05-18 12:10:57 +02:00
|
|
|
/*
|
|
|
|
* IF the file is unlinked, we cannot reopen
|
|
|
|
* the file later. So don't reclaim fd
|
|
|
|
*/
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_mark_fids_unreclaim(pdu, &fidp->path);
|
2011-05-18 12:10:57 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out_err;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_remove(pdu, &fidp->path);
|
2011-08-08 20:20:20 +02:00
|
|
|
if (!err) {
|
|
|
|
err = offset;
|
|
|
|
}
|
2011-05-18 12:10:57 +02:00
|
|
|
out_err:
|
2011-08-08 20:20:20 +02:00
|
|
|
/* For TREMOVE we need to clunk the fid even on failed remove */
|
2011-05-18 14:08:07 +02:00
|
|
|
clunk_fid(pdu->s, fidp->fid);
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-08-08 20:20:20 +02:00
|
|
|
complete_pdu(pdu->s, pdu, err);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-09-09 11:37:01 +02:00
|
|
|
static void v9fs_unlinkat(void *opaque)
|
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
V9fsString name;
|
|
|
|
int32_t dfid, flags;
|
|
|
|
size_t offset = 7;
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsPath path;
|
2011-09-09 11:37:01 +02:00
|
|
|
V9fsFidState *dfidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
|
|
|
|
pdu_unmarshal(pdu, offset, "dsd", &dfid, &name, &flags);
|
2011-10-12 15:41:24 +02:00
|
|
|
flags = dotl_to_at_flags(flags);
|
2011-09-09 11:37:01 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
dfidp = get_fid(pdu, dfid);
|
2011-09-09 11:37:01 +02:00
|
|
|
if (dfidp == NULL) {
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out_nofid;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* IF the file is unlinked, we cannot reopen
|
|
|
|
* the file later. So don't reclaim fd
|
|
|
|
*/
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_init(&path);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_name_to_path(pdu, &dfidp->path, name.data, &path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out_err;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_mark_fids_unreclaim(pdu, &path);
|
2011-09-09 11:37:01 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out_err;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_unlinkat(pdu, &dfidp->path, &name, flags);
|
2011-09-09 11:37:01 +02:00
|
|
|
if (!err) {
|
|
|
|
err = offset;
|
|
|
|
}
|
|
|
|
out_err:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, dfidp);
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&path);
|
2011-09-09 11:37:01 +02:00
|
|
|
out_nofid:
|
|
|
|
complete_pdu(pdu->s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
|
|
|
}
|
|
|
|
|
2011-09-09 11:44:18 +02:00
|
|
|
|
|
|
|
/* Only works with path name based fid */
|
2011-08-02 08:06:17 +02:00
|
|
|
static int v9fs_complete_rename(V9fsPDU *pdu, V9fsFidState *fidp,
|
2011-05-07 12:31:33 +02:00
|
|
|
int32_t newdirfid, V9fsString *name)
|
2010-04-29 14:15:00 +02:00
|
|
|
{
|
2011-05-07 12:31:33 +02:00
|
|
|
char *end;
|
2010-06-22 08:59:41 +02:00
|
|
|
int err = 0;
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsPath new_path;
|
|
|
|
V9fsFidState *tfidp;
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsState *s = pdu->s;
|
2011-05-18 14:08:07 +02:00
|
|
|
V9fsFidState *dirfidp = NULL;
|
2010-06-22 08:59:41 +02:00
|
|
|
char *old_name, *new_name;
|
2010-04-29 14:15:00 +02:00
|
|
|
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_init(&new_path);
|
2011-05-07 12:31:33 +02:00
|
|
|
if (newdirfid != -1) {
|
2011-08-02 08:06:17 +02:00
|
|
|
dirfidp = get_fid(pdu, newdirfid);
|
2010-06-22 08:59:41 +02:00
|
|
|
if (dirfidp == NULL) {
|
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-06-22 08:59:41 +02:00
|
|
|
}
|
2010-09-02 07:39:06 +02:00
|
|
|
BUG_ON(dirfidp->fid_type != P9_FID_NONE);
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_name_to_path(pdu, &dirfidp->path, name->data, &new_path);
|
2010-06-22 08:59:41 +02:00
|
|
|
} else {
|
2011-05-07 12:31:33 +02:00
|
|
|
old_name = fidp->path.data;
|
2010-04-29 14:15:00 +02:00
|
|
|
end = strrchr(old_name, '/');
|
|
|
|
if (end) {
|
|
|
|
end++;
|
|
|
|
} else {
|
|
|
|
end = old_name;
|
|
|
|
}
|
2011-08-21 05:09:37 +02:00
|
|
|
new_name = g_malloc0(end - old_name + name->size + 1);
|
2010-06-22 08:59:41 +02:00
|
|
|
strncat(new_name, old_name, end - old_name);
|
2011-05-07 12:31:33 +02:00
|
|
|
strncat(new_name + (end - old_name), name->data, name->size);
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_name_to_path(pdu, NULL, new_name, &new_path);
|
2011-09-09 11:44:18 +02:00
|
|
|
g_free(new_name);
|
2010-06-22 08:59:41 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_rename(pdu, &fidp->path, &new_path);
|
2011-09-09 11:44:18 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Fixup fid's pointing to the old name to
|
|
|
|
* start pointing to the new name
|
|
|
|
*/
|
|
|
|
for (tfidp = s->fid_list; tfidp; tfidp = tfidp->next) {
|
|
|
|
if (v9fs_path_is_ancestor(&fidp->path, &tfidp->path)) {
|
|
|
|
/* replace the name */
|
|
|
|
v9fs_fix_path(&tfidp->path, &new_path, strlen(fidp->path.data));
|
2010-04-29 14:15:00 +02:00
|
|
|
}
|
|
|
|
}
|
2010-06-22 08:59:41 +02:00
|
|
|
out:
|
2011-05-18 14:08:07 +02:00
|
|
|
if (dirfidp) {
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, dirfidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
}
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_free(&new_path);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2010-06-22 08:59:41 +02:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:05:54 +02:00
|
|
|
/* Only works with path name based fid */
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_rename(void *opaque)
|
2010-06-22 08:59:41 +02:00
|
|
|
{
|
|
|
|
int32_t fid;
|
|
|
|
ssize_t err = 0;
|
2011-05-07 12:31:33 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
V9fsString name;
|
|
|
|
int32_t newdirfid;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-06-22 08:59:41 +02:00
|
|
|
|
2011-05-07 12:31:33 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dds", &fid, &newdirfid, &name);
|
2010-06-22 08:59:41 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-07 12:31:33 +02:00
|
|
|
if (fidp == NULL) {
|
2010-06-22 08:59:41 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-06-22 08:59:41 +02:00
|
|
|
}
|
2011-05-07 12:31:33 +02:00
|
|
|
BUG_ON(fidp->fid_type != P9_FID_NONE);
|
2011-08-02 08:05:54 +02:00
|
|
|
/* if fs driver is not path based, return EOPNOTSUPP */
|
2011-10-12 17:29:18 +02:00
|
|
|
if (!(pdu->s->ctx.export_flags & V9FS_PATHNAME_FSCONTEXT)) {
|
2011-08-02 08:05:54 +02:00
|
|
|
err = -EOPNOTSUPP;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9fs_path_write_lock(s);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_complete_rename(pdu, fidp, newdirfid, &name);
|
2011-08-02 08:05:54 +02:00
|
|
|
v9fs_path_unlock(s);
|
2011-05-07 12:31:33 +02:00
|
|
|
if (!err) {
|
|
|
|
err = offset;
|
|
|
|
}
|
2011-08-02 08:05:54 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-05-07 12:31:33 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
2010-06-22 08:59:41 +02:00
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static void v9fs_fix_fid_paths(V9fsPDU *pdu, V9fsPath *olddir,
|
2011-09-09 11:44:18 +02:00
|
|
|
V9fsString *old_name, V9fsPath *newdir,
|
|
|
|
V9fsString *new_name)
|
|
|
|
{
|
|
|
|
V9fsFidState *tfidp;
|
|
|
|
V9fsPath oldpath, newpath;
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsState *s = pdu->s;
|
2011-09-09 11:44:18 +02:00
|
|
|
|
|
|
|
|
|
|
|
v9fs_path_init(&oldpath);
|
|
|
|
v9fs_path_init(&newpath);
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_co_name_to_path(pdu, olddir, old_name->data, &oldpath);
|
|
|
|
v9fs_co_name_to_path(pdu, newdir, new_name->data, &newpath);
|
2011-09-09 11:44:18 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Fixup fid's pointing to the old name to
|
|
|
|
* start pointing to the new name
|
|
|
|
*/
|
|
|
|
for (tfidp = s->fid_list; tfidp; tfidp = tfidp->next) {
|
|
|
|
if (v9fs_path_is_ancestor(&oldpath, &tfidp->path)) {
|
|
|
|
/* replace the name */
|
|
|
|
v9fs_fix_path(&tfidp->path, &newpath, strlen(oldpath.data));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
v9fs_path_free(&oldpath);
|
|
|
|
v9fs_path_free(&newpath);
|
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
static int v9fs_complete_renameat(V9fsPDU *pdu, int32_t olddirfid,
|
2011-05-23 19:54:41 +02:00
|
|
|
V9fsString *old_name, int32_t newdirfid,
|
|
|
|
V9fsString *new_name)
|
|
|
|
{
|
|
|
|
int err = 0;
|
2011-08-02 08:06:17 +02:00
|
|
|
V9fsState *s = pdu->s;
|
2011-05-23 19:54:41 +02:00
|
|
|
V9fsFidState *newdirfidp = NULL, *olddirfidp = NULL;
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
olddirfidp = get_fid(pdu, olddirfid);
|
2011-05-23 19:54:41 +02:00
|
|
|
if (olddirfidp == NULL) {
|
|
|
|
err = -ENOENT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (newdirfid != -1) {
|
2011-08-02 08:06:17 +02:00
|
|
|
newdirfidp = get_fid(pdu, newdirfid);
|
2011-05-23 19:54:41 +02:00
|
|
|
if (newdirfidp == NULL) {
|
|
|
|
err = -ENOENT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
} else {
|
2011-08-02 08:06:17 +02:00
|
|
|
newdirfidp = get_fid(pdu, olddirfid);
|
2011-05-23 19:54:41 +02:00
|
|
|
}
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_renameat(pdu, &olddirfidp->path, old_name,
|
2011-09-09 11:44:18 +02:00
|
|
|
&newdirfidp->path, new_name);
|
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
2011-05-23 19:54:41 +02:00
|
|
|
}
|
2011-10-12 17:29:18 +02:00
|
|
|
if (s->ctx.export_flags & V9FS_PATHNAME_FSCONTEXT) {
|
2011-08-02 08:05:54 +02:00
|
|
|
/* Only for path based fid we need to do the below fixup */
|
2011-08-02 08:06:17 +02:00
|
|
|
v9fs_fix_fid_paths(pdu, &olddirfidp->path, old_name,
|
2011-08-02 08:05:54 +02:00
|
|
|
&newdirfidp->path, new_name);
|
|
|
|
}
|
2011-05-23 19:54:41 +02:00
|
|
|
out:
|
|
|
|
if (olddirfidp) {
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, olddirfidp);
|
2011-05-23 19:54:41 +02:00
|
|
|
}
|
|
|
|
if (newdirfidp) {
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, newdirfidp);
|
2011-05-23 19:54:41 +02:00
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void v9fs_renameat(void *opaque)
|
|
|
|
{
|
|
|
|
ssize_t err = 0;
|
|
|
|
size_t offset = 7;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
|
|
|
int32_t olddirfid, newdirfid;
|
|
|
|
V9fsString old_name, new_name;
|
|
|
|
|
|
|
|
pdu_unmarshal(pdu, offset, "dsds", &olddirfid,
|
|
|
|
&old_name, &newdirfid, &new_name);
|
|
|
|
|
2011-08-02 08:05:54 +02:00
|
|
|
v9fs_path_write_lock(s);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_complete_renameat(pdu, olddirfid,
|
|
|
|
&old_name, newdirfid, &new_name);
|
2011-08-02 08:05:54 +02:00
|
|
|
v9fs_path_unlock(s);
|
2011-05-23 19:54:41 +02:00
|
|
|
if (!err) {
|
|
|
|
err = offset;
|
|
|
|
}
|
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
v9fs_string_free(&old_name);
|
|
|
|
v9fs_string_free(&new_name);
|
|
|
|
}
|
|
|
|
|
2011-05-08 09:36:04 +02:00
|
|
|
static void v9fs_wstat(void *opaque)
|
2010-04-29 14:15:00 +02:00
|
|
|
{
|
2011-05-08 09:36:04 +02:00
|
|
|
int32_t fid;
|
|
|
|
int err = 0;
|
|
|
|
int16_t unused;
|
|
|
|
V9fsStat v9stat;
|
|
|
|
size_t offset = 7;
|
|
|
|
struct stat stbuf;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-04-29 14:15:00 +02:00
|
|
|
|
2011-05-08 09:36:04 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dwS", &fid, &unused, &v9stat);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_wstat(pdu->tag, pdu->id, fid,
|
|
|
|
v9stat.mode, v9stat.atime, v9stat.mtime);
|
2011-05-18 14:08:07 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-08 09:36:04 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-04-29 14:15:00 +02:00
|
|
|
}
|
2011-05-08 09:36:04 +02:00
|
|
|
/* do we need to sync the file? */
|
|
|
|
if (donttouch_stat(&v9stat)) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_fsync(pdu, fidp, 0);
|
2010-04-29 14:15:00 +02:00
|
|
|
goto out;
|
|
|
|
}
|
2011-05-08 09:36:04 +02:00
|
|
|
if (v9stat.mode != -1) {
|
|
|
|
uint32_t v9_mode;
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lstat(pdu, &fidp->path, &stbuf);
|
2011-05-08 09:36:04 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
v9_mode = stat_to_v9mode(&stbuf);
|
|
|
|
if ((v9stat.mode & P9_STAT_MODE_TYPE_BITS) !=
|
|
|
|
(v9_mode & P9_STAT_MODE_TYPE_BITS)) {
|
|
|
|
/* Attempting to change the type */
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_chmod(pdu, &fidp->path,
|
2011-05-08 09:36:04 +02:00
|
|
|
v9mode_to_mode(v9stat.mode,
|
|
|
|
&v9stat.extension));
|
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (v9stat.mtime != -1 || v9stat.atime != -1) {
|
2010-06-09 15:44:38 +02:00
|
|
|
struct timespec times[2];
|
2011-05-08 09:36:04 +02:00
|
|
|
if (v9stat.atime != -1) {
|
|
|
|
times[0].tv_sec = v9stat.atime;
|
2010-06-09 15:44:38 +02:00
|
|
|
times[0].tv_nsec = 0;
|
|
|
|
} else {
|
|
|
|
times[0].tv_nsec = UTIME_OMIT;
|
|
|
|
}
|
2011-05-08 09:36:04 +02:00
|
|
|
if (v9stat.mtime != -1) {
|
|
|
|
times[1].tv_sec = v9stat.mtime;
|
2010-06-09 15:44:38 +02:00
|
|
|
times[1].tv_nsec = 0;
|
|
|
|
} else {
|
|
|
|
times[1].tv_nsec = UTIME_OMIT;
|
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_utimensat(pdu, &fidp->path, times);
|
2011-05-08 09:36:04 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
2010-04-29 14:15:00 +02:00
|
|
|
}
|
|
|
|
}
|
2011-05-08 09:36:04 +02:00
|
|
|
if (v9stat.n_gid != -1 || v9stat.n_uid != -1) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_chown(pdu, &fidp->path, v9stat.n_uid, v9stat.n_gid);
|
2011-05-08 09:36:04 +02:00
|
|
|
if (err < 0) {
|
2010-04-29 14:15:00 +02:00
|
|
|
goto out;
|
2011-05-08 09:36:04 +02:00
|
|
|
}
|
2010-04-29 14:15:00 +02:00
|
|
|
}
|
2011-05-08 09:36:04 +02:00
|
|
|
if (v9stat.name.size != 0) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_complete_rename(pdu, fidp, -1, &v9stat.name);
|
2011-05-08 09:36:04 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2010-04-29 14:15:00 +02:00
|
|
|
}
|
2011-05-08 09:36:04 +02:00
|
|
|
if (v9stat.length != -1) {
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_truncate(pdu, &fidp->path, v9stat.length);
|
2011-05-08 09:36:04 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2010-04-29 14:15:00 +02:00
|
|
|
}
|
2011-05-08 09:36:04 +02:00
|
|
|
err = offset;
|
2010-04-29 14:15:00 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-05-08 09:36:04 +02:00
|
|
|
v9fs_stat_free(&v9stat);
|
|
|
|
complete_pdu(s, pdu, err);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-05-19 01:04:01 +02:00
|
|
|
static int v9fs_fill_statfs(V9fsState *s, V9fsPDU *pdu, struct statfs *stbuf)
|
|
|
|
{
|
|
|
|
uint32_t f_type;
|
|
|
|
uint32_t f_bsize;
|
|
|
|
uint64_t f_blocks;
|
|
|
|
uint64_t f_bfree;
|
|
|
|
uint64_t f_bavail;
|
|
|
|
uint64_t f_files;
|
|
|
|
uint64_t f_ffree;
|
|
|
|
uint64_t fsid_val;
|
|
|
|
uint32_t f_namelen;
|
|
|
|
size_t offset = 7;
|
2010-06-09 15:44:28 +02:00
|
|
|
int32_t bsize_factor;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* compute bsize factor based on host file system block size
|
|
|
|
* and client msize
|
|
|
|
*/
|
2011-05-19 01:04:01 +02:00
|
|
|
bsize_factor = (s->msize - P9_IOHDRSZ)/stbuf->f_bsize;
|
2010-06-09 15:44:28 +02:00
|
|
|
if (!bsize_factor) {
|
|
|
|
bsize_factor = 1;
|
|
|
|
}
|
2011-05-19 01:04:01 +02:00
|
|
|
f_type = stbuf->f_type;
|
|
|
|
f_bsize = stbuf->f_bsize;
|
|
|
|
f_bsize *= bsize_factor;
|
2010-06-09 15:44:28 +02:00
|
|
|
/*
|
|
|
|
* f_bsize is adjusted(multiplied) by bsize factor, so we need to
|
|
|
|
* adjust(divide) the number of blocks, free blocks and available
|
|
|
|
* blocks by bsize factor
|
|
|
|
*/
|
2011-05-19 01:04:01 +02:00
|
|
|
f_blocks = stbuf->f_blocks/bsize_factor;
|
|
|
|
f_bfree = stbuf->f_bfree/bsize_factor;
|
|
|
|
f_bavail = stbuf->f_bavail/bsize_factor;
|
|
|
|
f_files = stbuf->f_files;
|
|
|
|
f_ffree = stbuf->f_ffree;
|
|
|
|
fsid_val = (unsigned int) stbuf->f_fsid.__val[0] |
|
|
|
|
(unsigned long long)stbuf->f_fsid.__val[1] << 32;
|
|
|
|
f_namelen = stbuf->f_namelen;
|
2010-05-10 08:41:03 +02:00
|
|
|
|
2011-05-19 01:04:01 +02:00
|
|
|
return pdu_marshal(pdu, offset, "ddqqqqqqd",
|
|
|
|
f_type, f_bsize, f_blocks, f_bfree,
|
|
|
|
f_bavail, f_files, f_ffree,
|
|
|
|
fsid_val, f_namelen);
|
2010-05-10 08:41:03 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_statfs(void *opaque)
|
2010-05-10 08:41:03 +02:00
|
|
|
{
|
2011-05-19 01:04:01 +02:00
|
|
|
int32_t fid;
|
|
|
|
ssize_t retval = 0;
|
|
|
|
size_t offset = 7;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
struct statfs stbuf;
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-05-10 08:41:03 +02:00
|
|
|
|
2011-05-19 01:04:01 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "d", &fid);
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-19 01:04:01 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
retval = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-05-10 08:41:03 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
retval = v9fs_co_statfs(pdu, &fidp->path, &stbuf);
|
2011-05-19 01:04:01 +02:00
|
|
|
if (retval < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
retval = offset;
|
|
|
|
retval += v9fs_fill_statfs(s, pdu, &stbuf);
|
2010-05-10 08:41:03 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-05-19 01:04:01 +02:00
|
|
|
complete_pdu(s, pdu, retval);
|
2011-05-18 23:18:05 +02:00
|
|
|
return;
|
2010-05-10 08:41:03 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_mknod(void *opaque)
|
2010-06-22 08:54:09 +02:00
|
|
|
{
|
2011-05-19 01:06:51 +02:00
|
|
|
|
|
|
|
int mode;
|
|
|
|
gid_t gid;
|
2010-06-22 08:54:09 +02:00
|
|
|
int32_t fid;
|
2011-05-19 01:06:51 +02:00
|
|
|
V9fsQID qid;
|
2010-06-22 08:54:09 +02:00
|
|
|
int err = 0;
|
|
|
|
int major, minor;
|
2011-05-19 01:06:51 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
V9fsString name;
|
|
|
|
struct stat stbuf;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-06-22 08:54:09 +02:00
|
|
|
|
2011-05-19 01:06:51 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dsdddd", &fid, &name, &mode,
|
|
|
|
&major, &minor, &gid);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_mknod(pdu->tag, pdu->id, fid, mode, major, minor);
|
2010-06-22 08:54:09 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2010-06-22 08:54:09 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-06-22 08:54:09 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_mknod(pdu, fidp, &name, fidp->uid, gid,
|
2011-05-24 11:40:56 +02:00
|
|
|
makedev(major, minor), mode, &stbuf);
|
2011-05-19 01:06:51 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
stat_to_qid(&stbuf, &qid);
|
|
|
|
err = offset;
|
|
|
|
err += pdu_marshal(pdu, offset, "Q", &qid);
|
2010-06-22 08:54:09 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_mknod_return(pdu->tag, pdu->id, qid.type, qid.version, qid.path);
|
2011-05-19 01:06:51 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
2010-06-22 08:54:09 +02:00
|
|
|
}
|
|
|
|
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
/*
|
|
|
|
* Implement posix byte range locking code
|
|
|
|
* Server side handling of locking code is very simple, because 9p server in
|
|
|
|
* QEMU can handle only one client. And most of the lock handling
|
|
|
|
* (like conflict, merging) etc is done by the VFS layer itself, so no need to
|
|
|
|
* do any thing in * qemu 9p server side lock code path.
|
|
|
|
* So when a TLOCK request comes, always return success
|
|
|
|
*/
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_lock(void *opaque)
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
{
|
2011-08-22 05:44:04 +02:00
|
|
|
int8_t status;
|
|
|
|
V9fsFlock *flock;
|
|
|
|
size_t offset = 7;
|
|
|
|
struct stat stbuf;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
int32_t fid, err = 0;
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
|
2011-08-22 05:44:04 +02:00
|
|
|
flock = g_malloc(sizeof(*flock));
|
|
|
|
pdu_unmarshal(pdu, offset, "dbdqqds", &fid, &flock->type,
|
|
|
|
&flock->flags, &flock->start, &flock->length,
|
|
|
|
&flock->proc_id, &flock->client_id);
|
2011-10-12 15:41:25 +02:00
|
|
|
|
|
|
|
trace_v9fs_lock(pdu->tag, pdu->id, fid,
|
|
|
|
flock->type, flock->start, flock->length);
|
|
|
|
|
2011-08-22 05:44:04 +02:00
|
|
|
status = P9_LOCK_ERROR;
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
|
|
|
|
/* We support only block flag now (that too ignored currently) */
|
2011-08-22 05:44:04 +02:00
|
|
|
if (flock->flags & ~P9_LOCK_FLAGS_BLOCK) {
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-08-22 05:44:04 +02:00
|
|
|
if (fidp == NULL) {
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_fstat(pdu, fidp->fs.fd, &stbuf);
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-08-22 05:44:04 +02:00
|
|
|
status = P9_LOCK_SUCCESS;
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-08-22 05:44:04 +02:00
|
|
|
err = offset;
|
|
|
|
err += pdu_marshal(pdu, offset, "b", status);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_lock_return(pdu->tag, pdu->id, status);
|
2011-08-22 05:44:04 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
2011-08-08 08:49:51 +02:00
|
|
|
v9fs_string_free(&flock->client_id);
|
2011-08-22 05:44:04 +02:00
|
|
|
g_free(flock);
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
}
|
|
|
|
|
2010-09-07 23:06:52 +02:00
|
|
|
/*
|
|
|
|
* When a TGETLOCK request comes, always return success because all lock
|
|
|
|
* handling is done by client's VFS layer.
|
|
|
|
*/
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_getlock(void *opaque)
|
2010-09-07 23:06:52 +02:00
|
|
|
{
|
2011-05-07 13:51:38 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
struct stat stbuf;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
V9fsGetlock *glock;
|
|
|
|
int32_t fid, err = 0;
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-09-07 23:06:52 +02:00
|
|
|
|
2011-05-07 13:51:38 +02:00
|
|
|
glock = g_malloc(sizeof(*glock));
|
|
|
|
pdu_unmarshal(pdu, offset, "dbqqds", &fid, &glock->type,
|
|
|
|
&glock->start, &glock->length, &glock->proc_id,
|
|
|
|
&glock->client_id);
|
2010-09-07 23:06:52 +02:00
|
|
|
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_getlock(pdu->tag, pdu->id, fid,
|
|
|
|
glock->type, glock->start, glock->length);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2011-05-07 13:51:38 +02:00
|
|
|
if (fidp == NULL) {
|
2010-09-07 23:06:52 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-09-07 23:06:52 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_fstat(pdu, fidp->fs.fd, &stbuf);
|
2010-09-07 23:06:52 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
2011-10-12 15:41:24 +02:00
|
|
|
glock->type = P9_LOCK_TYPE_UNLCK;
|
2011-05-07 13:51:38 +02:00
|
|
|
offset += pdu_marshal(pdu, offset, "bqqds", glock->type,
|
|
|
|
glock->start, glock->length, glock->proc_id,
|
|
|
|
&glock->client_id);
|
|
|
|
err = offset;
|
2010-09-07 23:06:52 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_getlock_return(pdu->tag, pdu->id, glock->type, glock->start,
|
|
|
|
glock->length, glock->proc_id);
|
|
|
|
|
2011-05-07 13:51:38 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
2011-08-08 08:49:51 +02:00
|
|
|
v9fs_string_free(&glock->client_id);
|
2011-05-07 13:51:38 +02:00
|
|
|
g_free(glock);
|
2010-09-07 23:06:52 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_mkdir(void *opaque)
|
2010-06-22 08:55:22 +02:00
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
2011-08-08 20:16:14 +02:00
|
|
|
size_t offset = 7;
|
2010-06-22 08:55:22 +02:00
|
|
|
int32_t fid;
|
2011-08-08 20:16:14 +02:00
|
|
|
struct stat stbuf;
|
|
|
|
V9fsQID qid;
|
2011-05-24 11:40:56 +02:00
|
|
|
V9fsString name;
|
2010-06-22 08:55:22 +02:00
|
|
|
V9fsFidState *fidp;
|
|
|
|
gid_t gid;
|
|
|
|
int mode;
|
2011-08-08 20:16:14 +02:00
|
|
|
int err = 0;
|
2010-06-22 08:55:22 +02:00
|
|
|
|
2011-08-08 20:16:14 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dsdd", &fid, &name, &mode, &gid);
|
2010-06-22 08:55:22 +02:00
|
|
|
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_mkdir(pdu->tag, pdu->id, fid, name.data, mode, gid);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2010-06-22 08:55:22 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-06-22 08:55:22 +02:00
|
|
|
}
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_mkdir(pdu, fidp, &name, mode, fidp->uid, gid, &stbuf);
|
2011-08-08 20:16:14 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
stat_to_qid(&stbuf, &qid);
|
|
|
|
offset += pdu_marshal(pdu, offset, "Q", &qid);
|
|
|
|
err = offset;
|
2010-06-22 08:55:22 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_mkdir_return(pdu->tag, pdu->id,
|
|
|
|
qid.type, qid.version, qid.path, err);
|
2011-08-08 20:16:14 +02:00
|
|
|
complete_pdu(pdu->s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
2010-06-22 08:55:22 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_xattrwalk(void *opaque)
|
2010-09-02 07:39:06 +02:00
|
|
|
{
|
2011-05-19 01:05:48 +02:00
|
|
|
int64_t size;
|
|
|
|
V9fsString name;
|
2010-09-02 07:39:06 +02:00
|
|
|
ssize_t err = 0;
|
2011-05-19 01:05:48 +02:00
|
|
|
size_t offset = 7;
|
2010-09-02 07:39:06 +02:00
|
|
|
int32_t fid, newfid;
|
2011-05-19 01:05:48 +02:00
|
|
|
V9fsFidState *file_fidp;
|
2011-05-18 14:08:07 +02:00
|
|
|
V9fsFidState *xattr_fidp = NULL;
|
2011-05-19 01:05:48 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-09-02 07:39:06 +02:00
|
|
|
|
2011-05-19 01:05:48 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dds", &fid, &newfid, &name);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_xattrwalk(pdu->tag, pdu->id, fid, newfid, name.data);
|
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
file_fidp = get_fid(pdu, fid);
|
2011-05-19 01:05:48 +02:00
|
|
|
if (file_fidp == NULL) {
|
2010-09-02 07:39:06 +02:00
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-09-02 07:39:06 +02:00
|
|
|
}
|
2011-05-19 01:05:48 +02:00
|
|
|
xattr_fidp = alloc_fid(s, newfid);
|
|
|
|
if (xattr_fidp == NULL) {
|
2010-09-02 07:39:06 +02:00
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2011-09-09 11:44:18 +02:00
|
|
|
v9fs_path_copy(&xattr_fidp->path, &file_fidp->path);
|
2011-05-19 01:05:48 +02:00
|
|
|
if (name.data[0] == 0) {
|
2010-09-02 07:39:06 +02:00
|
|
|
/*
|
|
|
|
* listxattr request. Get the size first
|
|
|
|
*/
|
2011-08-02 08:06:17 +02:00
|
|
|
size = v9fs_co_llistxattr(pdu, &xattr_fidp->path, NULL, 0);
|
2011-05-19 01:05:48 +02:00
|
|
|
if (size < 0) {
|
|
|
|
err = size;
|
2011-05-18 14:08:07 +02:00
|
|
|
clunk_fid(s, xattr_fidp->fid);
|
2011-05-19 01:05:48 +02:00
|
|
|
goto out;
|
2010-09-02 07:39:06 +02:00
|
|
|
}
|
2011-05-19 01:05:48 +02:00
|
|
|
/*
|
|
|
|
* Read the xattr value
|
|
|
|
*/
|
|
|
|
xattr_fidp->fs.xattr.len = size;
|
|
|
|
xattr_fidp->fid_type = P9_FID_XATTR;
|
|
|
|
xattr_fidp->fs.xattr.copied_len = -1;
|
|
|
|
if (size) {
|
2011-08-21 05:09:37 +02:00
|
|
|
xattr_fidp->fs.xattr.value = g_malloc(size);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_llistxattr(pdu, &xattr_fidp->path,
|
2011-05-19 01:05:48 +02:00
|
|
|
xattr_fidp->fs.xattr.value,
|
|
|
|
xattr_fidp->fs.xattr.len);
|
|
|
|
if (err < 0) {
|
2011-05-18 14:08:07 +02:00
|
|
|
clunk_fid(s, xattr_fidp->fid);
|
2011-05-19 01:05:48 +02:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
offset += pdu_marshal(pdu, offset, "q", size);
|
|
|
|
err = offset;
|
2010-09-02 07:39:06 +02:00
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* specific xattr fid. We check for xattr
|
|
|
|
* presence also collect the xattr size
|
|
|
|
*/
|
2011-08-02 08:06:17 +02:00
|
|
|
size = v9fs_co_lgetxattr(pdu, &xattr_fidp->path,
|
2011-05-19 01:05:48 +02:00
|
|
|
&name, NULL, 0);
|
|
|
|
if (size < 0) {
|
|
|
|
err = size;
|
2011-05-18 14:08:07 +02:00
|
|
|
clunk_fid(s, xattr_fidp->fid);
|
2011-05-19 01:05:48 +02:00
|
|
|
goto out;
|
2010-09-02 07:39:06 +02:00
|
|
|
}
|
2011-05-19 01:05:48 +02:00
|
|
|
/*
|
|
|
|
* Read the xattr value
|
|
|
|
*/
|
|
|
|
xattr_fidp->fs.xattr.len = size;
|
|
|
|
xattr_fidp->fid_type = P9_FID_XATTR;
|
|
|
|
xattr_fidp->fs.xattr.copied_len = -1;
|
|
|
|
if (size) {
|
2011-08-21 05:09:37 +02:00
|
|
|
xattr_fidp->fs.xattr.value = g_malloc(size);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_lgetxattr(pdu, &xattr_fidp->path,
|
2011-05-19 01:05:48 +02:00
|
|
|
&name, xattr_fidp->fs.xattr.value,
|
|
|
|
xattr_fidp->fs.xattr.len);
|
|
|
|
if (err < 0) {
|
2011-05-18 14:08:07 +02:00
|
|
|
clunk_fid(s, xattr_fidp->fid);
|
2011-05-19 01:05:48 +02:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
offset += pdu_marshal(pdu, offset, "q", size);
|
|
|
|
err = offset;
|
2010-09-02 07:39:06 +02:00
|
|
|
}
|
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, file_fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
if (xattr_fidp) {
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, xattr_fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
}
|
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_xattrwalk_return(pdu->tag, pdu->id, size);
|
2011-05-19 01:05:48 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
2010-09-02 07:39:06 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_xattrcreate(void *opaque)
|
2010-09-02 07:39:07 +02:00
|
|
|
{
|
|
|
|
int flags;
|
|
|
|
int32_t fid;
|
2011-05-19 01:06:26 +02:00
|
|
|
int64_t size;
|
2010-09-02 07:39:07 +02:00
|
|
|
ssize_t err = 0;
|
2011-05-19 01:06:26 +02:00
|
|
|
V9fsString name;
|
|
|
|
size_t offset = 7;
|
|
|
|
V9fsFidState *file_fidp;
|
|
|
|
V9fsFidState *xattr_fidp;
|
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
V9fsState *s = pdu->s;
|
2010-09-02 07:39:07 +02:00
|
|
|
|
2011-05-19 01:06:26 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "dsqd",
|
|
|
|
&fid, &name, &size, &flags);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_xattrcreate(pdu->tag, pdu->id, fid, name.data, size, flags);
|
2010-09-02 07:39:07 +02:00
|
|
|
|
2011-08-02 08:06:17 +02:00
|
|
|
file_fidp = get_fid(pdu, fid);
|
2011-05-19 01:06:26 +02:00
|
|
|
if (file_fidp == NULL) {
|
2010-09-02 07:39:07 +02:00
|
|
|
err = -EINVAL;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-09-02 07:39:07 +02:00
|
|
|
}
|
|
|
|
/* Make the file fid point to xattr */
|
2011-05-19 01:06:26 +02:00
|
|
|
xattr_fidp = file_fidp;
|
|
|
|
xattr_fidp->fid_type = P9_FID_XATTR;
|
|
|
|
xattr_fidp->fs.xattr.copied_len = 0;
|
|
|
|
xattr_fidp->fs.xattr.len = size;
|
|
|
|
xattr_fidp->fs.xattr.flags = flags;
|
|
|
|
v9fs_string_init(&xattr_fidp->fs.xattr.name);
|
|
|
|
v9fs_string_copy(&xattr_fidp->fs.xattr.name, &name);
|
|
|
|
if (size) {
|
2011-08-21 05:09:37 +02:00
|
|
|
xattr_fidp->fs.xattr.value = g_malloc(size);
|
2011-05-19 01:06:26 +02:00
|
|
|
} else {
|
|
|
|
xattr_fidp->fs.xattr.value = NULL;
|
|
|
|
}
|
|
|
|
err = offset;
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, file_fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-05-19 01:06:26 +02:00
|
|
|
complete_pdu(s, pdu, err);
|
|
|
|
v9fs_string_free(&name);
|
2010-09-02 07:39:07 +02:00
|
|
|
}
|
2010-09-02 07:39:06 +02:00
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_readlink(void *opaque)
|
2010-09-14 11:38:25 +02:00
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
2011-08-08 20:06:41 +02:00
|
|
|
size_t offset = 7;
|
|
|
|
V9fsString target;
|
2010-09-14 11:38:25 +02:00
|
|
|
int32_t fid;
|
|
|
|
int err = 0;
|
|
|
|
V9fsFidState *fidp;
|
|
|
|
|
2011-08-08 20:06:41 +02:00
|
|
|
pdu_unmarshal(pdu, offset, "d", &fid);
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_readlink(pdu->tag, pdu->id, fid);
|
2011-08-02 08:06:17 +02:00
|
|
|
fidp = get_fid(pdu, fid);
|
2010-09-14 11:38:25 +02:00
|
|
|
if (fidp == NULL) {
|
|
|
|
err = -ENOENT;
|
2011-05-18 14:08:07 +02:00
|
|
|
goto out_nofid;
|
2010-09-14 11:38:25 +02:00
|
|
|
}
|
|
|
|
|
2011-08-08 20:06:41 +02:00
|
|
|
v9fs_string_init(&target);
|
2011-08-02 08:06:17 +02:00
|
|
|
err = v9fs_co_readlink(pdu, &fidp->path, &target);
|
2011-08-08 20:06:41 +02:00
|
|
|
if (err < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
offset += pdu_marshal(pdu, offset, "s", &target);
|
|
|
|
err = offset;
|
|
|
|
v9fs_string_free(&target);
|
2010-09-14 11:38:25 +02:00
|
|
|
out:
|
2011-08-02 08:06:17 +02:00
|
|
|
put_fid(pdu, fidp);
|
2011-05-18 14:08:07 +02:00
|
|
|
out_nofid:
|
2011-10-12 15:41:25 +02:00
|
|
|
trace_v9fs_readlink_return(pdu->tag, pdu->id, target.data);
|
2011-08-08 20:06:41 +02:00
|
|
|
complete_pdu(pdu->s, pdu, err);
|
2010-09-14 11:38:25 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static CoroutineEntry *pdu_co_handlers[] = {
|
2010-06-09 11:27:57 +02:00
|
|
|
[P9_TREADDIR] = v9fs_readdir,
|
2010-05-10 08:41:03 +02:00
|
|
|
[P9_TSTATFS] = v9fs_statfs,
|
virtio-9p: getattr server implementation for 9P2000.L protocol.
SYNOPSIS
size[4] Tgetattr tag[2] fid[4] request_mask[8]
size[4] Rgetattr tag[2] lstat[n]
DESCRIPTION
The getattr transaction inquires about the file identified by fid.
request_mask is a bit mask that specifies which fields of the
stat structure is the client interested in.
The reply will contain a machine-independent directory entry,
laid out as follows:
st_result_mask[8]
Bit mask that indicates which fields in the stat structure
have been populated by the server
qid.type[1]
the type of the file (directory, etc.), represented as a bit
vector corresponding to the high 8 bits of the file's mode
word.
qid.vers[4]
version number for given path
qid.path[8]
the file server's unique identification for the file
st_mode[4]
Permission and flags
st_uid[4]
User id of owner
st_gid[4]
Group ID of owner
st_nlink[8]
Number of hard links
st_rdev[8]
Device ID (if special file)
st_size[8]
Size, in bytes
st_blksize[8]
Block size for file system IO
st_blocks[8]
Number of file system blocks allocated
st_atime_sec[8]
Time of last access, seconds
st_atime_nsec[8]
Time of last access, nanoseconds
st_mtime_sec[8]
Time of last modification, seconds
st_mtime_nsec[8]
Time of last modification, nanoseconds
st_ctime_sec[8]
Time of last status change, seconds
st_ctime_nsec[8]
Time of last status change, nanoseconds
st_btime_sec[8]
Time of creation (birth) of file, seconds
st_btime_nsec[8]
Time of creation (birth) of file, nanoseconds
st_gen[8]
Inode generation
st_data_version[8]
Data version number
request_mask and result_mask bit masks contain the following bits
#define P9_STATS_MODE 0x00000001ULL
#define P9_STATS_NLINK 0x00000002ULL
#define P9_STATS_UID 0x00000004ULL
#define P9_STATS_GID 0x00000008ULL
#define P9_STATS_RDEV 0x00000010ULL
#define P9_STATS_ATIME 0x00000020ULL
#define P9_STATS_MTIME 0x00000040ULL
#define P9_STATS_CTIME 0x00000080ULL
#define P9_STATS_INO 0x00000100ULL
#define P9_STATS_SIZE 0x00000200ULL
#define P9_STATS_BLOCKS 0x00000400ULL
#define P9_STATS_BTIME 0x00000800ULL
#define P9_STATS_GEN 0x00001000ULL
#define P9_STATS_DATA_VERSION 0x00002000ULL
#define P9_STATS_BASIC 0x000007ffULL
#define P9_STATS_ALL 0x00003fffULL
This patch implements the client side of getattr implementation for 9P2000.L.
It introduces a new structure p9_stat_dotl for getting Linux stat information
along with QID. The data layout is similar to stat structure in Linux user
space with the following major differences:
inode (st_ino) is not part of data. Instead qid is.
device (st_dev) is not part of data because this doesn't make sense on the
client.
All time variables are 64 bit wide on the wire. The kernel seems to use
32 bit variables for these variables. However, some of the architectures
have used 64 bit variables and glibc exposes 64 bit variables to user
space on some architectures. Hence to be on the safer side we have made
these 64 bit in the protocol. Refer to the comments in
include/asm-generic/stat.h
There are some additional fields: st_btime_sec, st_btime_nsec, st_gen,
st_data_version apart from the bitmask, st_result_mask. The bit mask
is filled by the server to indicate which stat fields have been
populated by the server. Currently there is no clean way for the
server to obtain these additional fields, so it sends back just the
basic fields.
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
2010-07-20 08:14:41 +02:00
|
|
|
[P9_TGETATTR] = v9fs_getattr,
|
virtio-9p: Implement server side of setattr for 9P2000.L protocol.
SYNOPSIS
size[4] Tsetattr tag[2] attr[n]
size[4] Rsetattr tag[2]
DESCRIPTION
The setattr command changes some of the file status information.
attr resembles the iattr structure used in Linux kernel. It
specifies which status parameter is to be changed and to what
value. It is laid out as follows:
valid[4]
specifies which status information is to be changed. Possible
values are:
ATTR_MODE (1 << 0)
ATTR_UID (1 << 1)
ATTR_GID (1 << 2)
ATTR_SIZE (1 << 3)
ATTR_ATIME (1 << 4)
ATTR_MTIME (1 << 5)
ATTR_CTIME (1 << 5)
ATTR_ATIME_SET (1 << 7)
ATTR_MTIME_SET (1 << 8)
The last two bits represent whether the time information
is being sent by the client's user space. In the absense
of these bits the server always uses server's time.
mode[4]
File permission bits
uid[4]
Owner id of file
gid[4]
Group id of the file
size[8]
File size
atime_sec[8]
Time of last file access, seconds
atime_nsec[8]
Time of last file access, nanoseconds
mtime_sec[8]
Time of last file modification, seconds
mtime_nsec[8]
Time of last file modification, nanoseconds
Explanation of the patches:
--------------------------
*) The kernel just copies relevent contents of iattr structure to p9_iattr_dotl
structure and passes it down to the client. The only check it has is calling
inode_change_ok()
*) The p9_iattr_dotl structure does not have ctime and ia_file parameters because
I don't think these are needed in our case. The client user space can request
updating just ctime by calling chown(fd, -1, -1). This is handled on server
side without a need for putting ctime on the wire.
*) The server currently supports changing mode, time, ownership and size of the
file.
*) 9P RFC says "Either all the changes in wstat request happen, or none of them
does: if the request succeeds, all changes were made; if it fails, none were."
I have not done anything to implement this specifically because I don't see
a reason.
[jvrao@linux.vnet.ibm.com: Parts of code for handling chown(-1,-1)
Signed-off-by: Sripathi Kodi <sripathik@in.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-06-17 14:48:47 +02:00
|
|
|
[P9_TSETATTR] = v9fs_setattr,
|
2010-09-02 07:39:06 +02:00
|
|
|
[P9_TXATTRWALK] = v9fs_xattrwalk,
|
2010-09-02 07:39:07 +02:00
|
|
|
[P9_TXATTRCREATE] = v9fs_xattrcreate,
|
2010-06-22 08:54:09 +02:00
|
|
|
[P9_TMKNOD] = v9fs_mknod,
|
2010-06-22 08:59:41 +02:00
|
|
|
[P9_TRENAME] = v9fs_rename,
|
[virto-9p] Implement TLOCK
Synopsis
size[4] TLock tag[2] fid[4] flock[n]
size[4] RLock tag[2] status[1]
Description
Tlock is used to acquire/release byte range posix locks on a file
identified by given fid. The reply contains status of the lock request
flock structure:
type[1] - Type of lock: F_RDLCK, F_WRLCK, F_UNLCK
flags[4] - Flags could be either of
P9_LOCK_FLAGS_BLOCK(1) - Blocked lock request, if there is a
conflicting lock exists, wait for that lock to be released.
P9_LOCK_FLAGS_RECLAIM(2) - Reclaim lock request, used when client is
trying to reclaim a lock after a server restrart (due to crash)
start[8] - Starting offset for lock
length[8] - Number of bytes to lock
If length is 0, lock all bytes starting at the location 'start'
through to the end of file
pid[4] - PID of the process that wants to take lock
client_id[4] - Unique client id
status[1] - Status of the lock request, can be
P9_LOCK_SUCCESS(0), P9_LOCK_BLOCKED(1), P9_LOCK_ERROR(2) or
P9_LOCK_GRACE(3)
P9_LOCK_SUCCESS - Request was successful
P9_LOCK_BLOCKED - A conflicting lock is held by another process
P9_LOCK_ERROR - Error while processing the lock request
P9_LOCK_GRACE - Server is in grace period, it can't accept new lock
requests in this period (except locks with
P9_LOCK_FLAGS_RECLAIM flag set)
Signed-off-by: M. Mohan Kumar <mohan@in.ibm.com>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
2010-09-07 22:49:32 +02:00
|
|
|
[P9_TLOCK] = v9fs_lock,
|
2010-09-07 23:06:52 +02:00
|
|
|
[P9_TGETLOCK] = v9fs_getlock,
|
2011-05-23 19:54:41 +02:00
|
|
|
[P9_TRENAMEAT] = v9fs_renameat,
|
2010-09-14 11:38:25 +02:00
|
|
|
[P9_TREADLINK] = v9fs_readlink,
|
2011-09-09 11:37:01 +02:00
|
|
|
[P9_TUNLINKAT] = v9fs_unlinkat,
|
2010-06-22 08:55:22 +02:00
|
|
|
[P9_TMKDIR] = v9fs_mkdir,
|
2010-04-29 14:14:44 +02:00
|
|
|
[P9_TVERSION] = v9fs_version,
|
2010-06-22 16:17:04 +02:00
|
|
|
[P9_TLOPEN] = v9fs_open,
|
2010-04-29 14:14:44 +02:00
|
|
|
[P9_TATTACH] = v9fs_attach,
|
|
|
|
[P9_TSTAT] = v9fs_stat,
|
|
|
|
[P9_TWALK] = v9fs_walk,
|
|
|
|
[P9_TCLUNK] = v9fs_clunk,
|
2010-09-23 02:18:33 +02:00
|
|
|
[P9_TFSYNC] = v9fs_fsync,
|
2010-04-29 14:14:44 +02:00
|
|
|
[P9_TOPEN] = v9fs_open,
|
|
|
|
[P9_TREAD] = v9fs_read,
|
|
|
|
#if 0
|
|
|
|
[P9_TAUTH] = v9fs_auth,
|
|
|
|
#endif
|
|
|
|
[P9_TFLUSH] = v9fs_flush,
|
2010-06-09 20:21:15 +02:00
|
|
|
[P9_TLINK] = v9fs_link,
|
2010-06-09 23:02:08 +02:00
|
|
|
[P9_TSYMLINK] = v9fs_symlink,
|
2010-04-29 14:14:44 +02:00
|
|
|
[P9_TCREATE] = v9fs_create,
|
2010-06-18 03:27:24 +02:00
|
|
|
[P9_TLCREATE] = v9fs_lcreate,
|
2010-04-29 14:14:44 +02:00
|
|
|
[P9_TWRITE] = v9fs_write,
|
|
|
|
[P9_TWSTAT] = v9fs_wstat,
|
|
|
|
[P9_TREMOVE] = v9fs_remove,
|
|
|
|
};
|
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
static void v9fs_op_not_supp(void *opaque)
|
2011-06-01 09:05:14 +02:00
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
V9fsPDU *pdu = opaque;
|
|
|
|
complete_pdu(pdu->s, pdu, -EOPNOTSUPP);
|
2011-06-01 09:05:14 +02:00
|
|
|
}
|
|
|
|
|
2010-04-29 14:14:44 +02:00
|
|
|
static void submit_pdu(V9fsState *s, V9fsPDU *pdu)
|
|
|
|
{
|
2011-05-18 23:18:05 +02:00
|
|
|
Coroutine *co;
|
|
|
|
CoroutineEntry *handler;
|
2010-04-29 14:14:44 +02:00
|
|
|
|
2011-05-18 23:18:05 +02:00
|
|
|
if (pdu->id >= ARRAY_SIZE(pdu_co_handlers) ||
|
|
|
|
(pdu_co_handlers[pdu->id] == NULL)) {
|
2011-06-01 09:05:14 +02:00
|
|
|
handler = v9fs_op_not_supp;
|
|
|
|
} else {
|
2011-05-18 23:18:05 +02:00
|
|
|
handler = pdu_co_handlers[pdu->id];
|
2011-06-01 09:05:14 +02:00
|
|
|
}
|
2011-05-18 23:18:05 +02:00
|
|
|
co = qemu_coroutine_create(handler);
|
|
|
|
qemu_coroutine_enter(co, pdu);
|
2010-04-29 14:14:44 +02:00
|
|
|
}
|
|
|
|
|
2011-06-01 09:05:13 +02:00
|
|
|
void handle_9p_output(VirtIODevice *vdev, VirtQueue *vq)
|
2010-04-29 14:14:44 +02:00
|
|
|
{
|
|
|
|
V9fsState *s = (V9fsState *)vdev;
|
|
|
|
V9fsPDU *pdu;
|
|
|
|
ssize_t len;
|
|
|
|
|
|
|
|
while ((pdu = alloc_pdu(s)) &&
|
|
|
|
(len = virtqueue_pop(vq, &pdu->elem)) != 0) {
|
|
|
|
uint8_t *ptr;
|
2011-05-18 23:18:05 +02:00
|
|
|
pdu->s = s;
|
2010-04-29 14:14:44 +02:00
|
|
|
BUG_ON(pdu->elem.out_num == 0 || pdu->elem.in_num == 0);
|
|
|
|
BUG_ON(pdu->elem.out_sg[0].iov_len < 7);
|
|
|
|
|
|
|
|
ptr = pdu->elem.out_sg[0].iov_base;
|
|
|
|
|
|
|
|
memcpy(&pdu->size, ptr, 4);
|
|
|
|
pdu->id = ptr[4];
|
|
|
|
memcpy(&pdu->tag, ptr + 5, 2);
|
2011-08-02 08:06:17 +02:00
|
|
|
qemu_co_queue_init(&pdu->complete);
|
2010-04-29 14:14:44 +02:00
|
|
|
submit_pdu(s, pdu);
|
|
|
|
}
|
|
|
|
free_pdu(s, pdu);
|
|
|
|
}
|
2011-05-18 12:10:57 +02:00
|
|
|
|
|
|
|
void virtio_9p_set_fd_limit(void)
|
|
|
|
{
|
|
|
|
struct rlimit rlim;
|
|
|
|
if (getrlimit(RLIMIT_NOFILE, &rlim) < 0) {
|
|
|
|
fprintf(stderr, "Failed to get the resource limit\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
open_fd_hw = rlim.rlim_cur - MIN(400, rlim.rlim_cur/3);
|
|
|
|
open_fd_rc = rlim.rlim_cur/2;
|
|
|
|
}
|