virtio-9p: Add P9_TWSTAT support
Implement P9_TWSTAT support. This gets file and directory creation to work. [jvrao@linux.vnet.ibm.com: strdup to qemu_strdup conversion] [aneesh.kumar@linux.vnet.ibm.com: v9fs_fix_path] Signed-off-by: Anthony Liguori <aliguori@us.ibm.com> Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
This commit is contained in:
		
							parent
							
								
									c494dd6f28
								
							
						
					
					
						commit
						8cf89e007a
					
				| 
						 | 
				
			
			@ -30,8 +30,10 @@ typedef struct FileOperations
 | 
			
		|||
    int (*lstat)(FsContext *, const char *, struct stat *);
 | 
			
		||||
    ssize_t (*readlink)(FsContext *, const char *, char *, size_t);
 | 
			
		||||
    int (*chmod)(FsContext *, const char *, mode_t);
 | 
			
		||||
    int (*chown)(FsContext *, const char *, uid_t, gid_t);
 | 
			
		||||
    int (*mknod)(FsContext *, const char *, mode_t, dev_t);
 | 
			
		||||
    int (*mksock)(FsContext *, const char *);
 | 
			
		||||
    int (*utime)(FsContext *, const char *, const struct utimbuf *);
 | 
			
		||||
    int (*symlink)(FsContext *, const char *, const char *);
 | 
			
		||||
    int (*link)(FsContext *, const char *, const char *);
 | 
			
		||||
    int (*setuid)(FsContext *, uid_t);
 | 
			
		||||
| 
						 | 
				
			
			@ -49,6 +51,9 @@ typedef struct FileOperations
 | 
			
		|||
    off_t (*lseek)(FsContext *, int, off_t, int);
 | 
			
		||||
    int (*mkdir)(FsContext *, const char *, mode_t);
 | 
			
		||||
    int (*fstat)(FsContext *, int, struct stat *);
 | 
			
		||||
    int (*rename)(FsContext *, const char *, const char *);
 | 
			
		||||
    int (*truncate)(FsContext *, const char *, off_t);
 | 
			
		||||
    int (*fsync)(FsContext *, int);
 | 
			
		||||
    void *opaque;
 | 
			
		||||
} FileOperations;
 | 
			
		||||
#endif
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -212,6 +212,51 @@ static int local_link(FsContext *ctx, const char *oldpath, const char *newpath)
 | 
			
		|||
    return err;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int local_truncate(FsContext *ctx, const char *path, off_t size)
 | 
			
		||||
{
 | 
			
		||||
    return truncate(rpath(ctx, path), size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int local_rename(FsContext *ctx, const char *oldpath,
 | 
			
		||||
                        const char *newpath)
 | 
			
		||||
{
 | 
			
		||||
    char *tmp;
 | 
			
		||||
    int err;
 | 
			
		||||
 | 
			
		||||
    tmp = qemu_strdup(rpath(ctx, oldpath));
 | 
			
		||||
    if (tmp == NULL) {
 | 
			
		||||
        return -1;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    err = rename(tmp, rpath(ctx, newpath));
 | 
			
		||||
    if (err == -1) {
 | 
			
		||||
        int serrno = errno;
 | 
			
		||||
        qemu_free(tmp);
 | 
			
		||||
        errno = serrno;
 | 
			
		||||
    } else {
 | 
			
		||||
        qemu_free(tmp);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return err;
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int local_chown(FsContext *ctx, const char *path, uid_t uid, gid_t gid)
 | 
			
		||||
{
 | 
			
		||||
    return chown(rpath(ctx, path), uid, gid);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int local_utime(FsContext *ctx, const char *path,
 | 
			
		||||
                        const struct utimbuf *buf)
 | 
			
		||||
{
 | 
			
		||||
    return utime(rpath(ctx, path), buf);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int local_fsync(FsContext *ctx, int fd)
 | 
			
		||||
{
 | 
			
		||||
    return fsync(fd);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
FileOperations local_ops = {
 | 
			
		||||
    .lstat = local_lstat,
 | 
			
		||||
    .setuid = local_setuid,
 | 
			
		||||
| 
						 | 
				
			
			@ -235,4 +280,9 @@ FileOperations local_ops = {
 | 
			
		|||
    .open2 = local_open2,
 | 
			
		||||
    .symlink = local_symlink,
 | 
			
		||||
    .link = local_link,
 | 
			
		||||
    .truncate = local_truncate,
 | 
			
		||||
    .rename = local_rename,
 | 
			
		||||
    .chown = local_chown,
 | 
			
		||||
    .utime = local_utime,
 | 
			
		||||
    .fsync = local_fsync,
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										273
									
								
								hw/virtio-9p.c
								
								
								
								
							
							
						
						
									
										273
									
								
								hw/virtio-9p.c
								
								
								
								
							| 
						 | 
				
			
			@ -144,6 +144,33 @@ static int v9fs_do_link(V9fsState *s, V9fsString *oldpath, V9fsString *newpath)
 | 
			
		|||
    return s->ops->link(&s->ctx, oldpath->data, newpath->data);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int v9fs_do_truncate(V9fsState *s, V9fsString *path, off_t size)
 | 
			
		||||
{
 | 
			
		||||
    return s->ops->truncate(&s->ctx, path->data, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int v9fs_do_rename(V9fsState *s, V9fsString *oldpath,
 | 
			
		||||
                            V9fsString *newpath)
 | 
			
		||||
{
 | 
			
		||||
    return s->ops->rename(&s->ctx, oldpath->data, newpath->data);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int v9fs_do_chown(V9fsState *s, V9fsString *path, uid_t uid, gid_t gid)
 | 
			
		||||
{
 | 
			
		||||
    return s->ops->chown(&s->ctx, path->data, uid, gid);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int v9fs_do_utime(V9fsState *s, V9fsString *path,
 | 
			
		||||
                            const struct utimbuf *buf)
 | 
			
		||||
{
 | 
			
		||||
    return s->ops->utime(&s->ctx, path->data, buf);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int v9fs_do_fsync(V9fsState *s, int fd)
 | 
			
		||||
{
 | 
			
		||||
    return s->ops->fsync(&s->ctx, fd);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_string_init(V9fsString *str)
 | 
			
		||||
{
 | 
			
		||||
    str->data = NULL;
 | 
			
		||||
| 
						 | 
				
			
			@ -929,6 +956,15 @@ static void v9fs_dummy(V9fsState *s, V9fsPDU *pdu)
 | 
			
		|||
    (void) print_sg;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_fix_path(V9fsString *dst, V9fsString *src, int len)
 | 
			
		||||
{
 | 
			
		||||
    V9fsString str;
 | 
			
		||||
    v9fs_string_init(&str);
 | 
			
		||||
    v9fs_string_copy(&str, dst);
 | 
			
		||||
    v9fs_string_sprintf(dst, "%s%s", src->data, str.data+len);
 | 
			
		||||
    v9fs_string_free(&str);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_version(V9fsState *s, V9fsPDU *pdu)
 | 
			
		||||
{
 | 
			
		||||
    int32_t msize;
 | 
			
		||||
| 
						 | 
				
			
			@ -1931,11 +1967,244 @@ static void v9fs_remove(V9fsState *s, V9fsPDU *pdu)
 | 
			
		|||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
typedef struct V9fsWstatState
 | 
			
		||||
{
 | 
			
		||||
    V9fsPDU *pdu;
 | 
			
		||||
    size_t offset;
 | 
			
		||||
    int16_t unused;
 | 
			
		||||
    V9fsStat v9stat;
 | 
			
		||||
    V9fsFidState *fidp;
 | 
			
		||||
    struct stat stbuf;
 | 
			
		||||
    V9fsString nname;
 | 
			
		||||
} V9fsWstatState;
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat_post_truncate(V9fsState *s, V9fsWstatState *vs, int err)
 | 
			
		||||
{
 | 
			
		||||
    if (err < 0) {
 | 
			
		||||
        goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    err = vs->offset;
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat_post_rename(V9fsState *s, V9fsWstatState *vs, int err)
 | 
			
		||||
{
 | 
			
		||||
    if (err < 0) {
 | 
			
		||||
        goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (vs->v9stat.name.size != 0) {
 | 
			
		||||
        v9fs_string_free(&vs->nname);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (vs->v9stat.length != -1) {
 | 
			
		||||
        if (v9fs_do_truncate(s, &vs->fidp->path, vs->v9stat.length) < 0) {
 | 
			
		||||
            err = -errno;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    v9fs_wstat_post_truncate(s, vs, err);
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat_post_chown(V9fsState *s, V9fsWstatState *vs, int err)
 | 
			
		||||
{
 | 
			
		||||
    V9fsFidState *fidp;
 | 
			
		||||
    if (err < 0) {
 | 
			
		||||
        goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (vs->v9stat.name.size != 0) {
 | 
			
		||||
        char *old_name, *new_name;
 | 
			
		||||
        char *end;
 | 
			
		||||
 | 
			
		||||
        old_name = vs->fidp->path.data;
 | 
			
		||||
        end = strrchr(old_name, '/');
 | 
			
		||||
        if (end) {
 | 
			
		||||
            end++;
 | 
			
		||||
        } else {
 | 
			
		||||
            end = old_name;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        new_name = qemu_malloc(end - old_name + vs->v9stat.name.size + 1);
 | 
			
		||||
 | 
			
		||||
        memset(new_name, 0, end - old_name + vs->v9stat.name.size + 1);
 | 
			
		||||
        memcpy(new_name, old_name, end - old_name);
 | 
			
		||||
        memcpy(new_name + (end - old_name), vs->v9stat.name.data,
 | 
			
		||||
                vs->v9stat.name.size);
 | 
			
		||||
        vs->nname.data = new_name;
 | 
			
		||||
        vs->nname.size = strlen(new_name);
 | 
			
		||||
 | 
			
		||||
        if (strcmp(new_name, vs->fidp->path.data) != 0) {
 | 
			
		||||
            if (v9fs_do_rename(s, &vs->fidp->path, &vs->nname)) {
 | 
			
		||||
                err = -errno;
 | 
			
		||||
            } else {
 | 
			
		||||
                /*
 | 
			
		||||
                 * Fixup fid's pointing to the old name to
 | 
			
		||||
                 * start pointing to the new name
 | 
			
		||||
                 */
 | 
			
		||||
                for (fidp = s->fid_list; fidp; fidp = fidp->next) {
 | 
			
		||||
 | 
			
		||||
                    if (vs->fidp == fidp) {
 | 
			
		||||
                        /*
 | 
			
		||||
                         * we replace name of this fid towards the end
 | 
			
		||||
                         * so that our below strcmp will work
 | 
			
		||||
                         */
 | 
			
		||||
                        continue;
 | 
			
		||||
                    }
 | 
			
		||||
                    if (!strncmp(vs->fidp->path.data, fidp->path.data,
 | 
			
		||||
                                 strlen(vs->fidp->path.data))) {
 | 
			
		||||
                        /* replace the name */
 | 
			
		||||
                        v9fs_fix_path(&fidp->path, &vs->nname,
 | 
			
		||||
                                      strlen(vs->fidp->path.data));
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                v9fs_string_copy(&vs->fidp->path, &vs->nname);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    v9fs_wstat_post_rename(s, vs, err);
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat_post_utime(V9fsState *s, V9fsWstatState *vs, int err)
 | 
			
		||||
{
 | 
			
		||||
    if (err < 0) {
 | 
			
		||||
        goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (vs->v9stat.n_gid != -1) {
 | 
			
		||||
        if (v9fs_do_chown(s, &vs->fidp->path, vs->v9stat.n_uid,
 | 
			
		||||
                    vs->v9stat.n_gid)) {
 | 
			
		||||
            err = -errno;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    v9fs_wstat_post_chown(s, vs, err);
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat_post_chmod(V9fsState *s, V9fsWstatState *vs, int err)
 | 
			
		||||
{
 | 
			
		||||
    if (err < 0) {
 | 
			
		||||
        goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (vs->v9stat.mtime != -1) {
 | 
			
		||||
        struct utimbuf tb;
 | 
			
		||||
        tb.actime = 0;
 | 
			
		||||
        tb.modtime = vs->v9stat.mtime;
 | 
			
		||||
        if (v9fs_do_utime(s, &vs->fidp->path, &tb)) {
 | 
			
		||||
            err = -errno;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    v9fs_wstat_post_utime(s, vs, err);
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat_post_fsync(V9fsState *s, V9fsWstatState *vs, int err)
 | 
			
		||||
{
 | 
			
		||||
    if (err == -1) {
 | 
			
		||||
        err = -errno;
 | 
			
		||||
    }
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat_post_lstat(V9fsState *s, V9fsWstatState *vs, int err)
 | 
			
		||||
{
 | 
			
		||||
    uint32_t v9_mode;
 | 
			
		||||
 | 
			
		||||
    if (err == -1) {
 | 
			
		||||
        err = -errno;
 | 
			
		||||
        goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    v9_mode = stat_to_v9mode(&vs->stbuf);
 | 
			
		||||
 | 
			
		||||
    if ((vs->v9stat.mode & P9_STAT_MODE_TYPE_BITS) !=
 | 
			
		||||
        (v9_mode & P9_STAT_MODE_TYPE_BITS)) {
 | 
			
		||||
            /* Attempting to change the type */
 | 
			
		||||
            err = -EIO;
 | 
			
		||||
            goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (v9fs_do_chmod(s, &vs->fidp->path, v9mode_to_mode(vs->v9stat.mode,
 | 
			
		||||
                    &vs->v9stat.extension))) {
 | 
			
		||||
            err = -errno;
 | 
			
		||||
     }
 | 
			
		||||
    v9fs_wstat_post_chmod(s, vs, err);
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void v9fs_wstat(V9fsState *s, V9fsPDU *pdu)
 | 
			
		||||
{
 | 
			
		||||
    if (debug_9p_pdu) {
 | 
			
		||||
        pprint_pdu(pdu);
 | 
			
		||||
    int32_t fid;
 | 
			
		||||
    V9fsWstatState *vs;
 | 
			
		||||
    int err = 0;
 | 
			
		||||
 | 
			
		||||
    vs = qemu_malloc(sizeof(*vs));
 | 
			
		||||
    vs->pdu = pdu;
 | 
			
		||||
    vs->offset = 7;
 | 
			
		||||
 | 
			
		||||
    pdu_unmarshal(pdu, vs->offset, "dwS", &fid, &vs->unused, &vs->v9stat);
 | 
			
		||||
 | 
			
		||||
    vs->fidp = lookup_fid(s, fid);
 | 
			
		||||
    if (vs->fidp == NULL) {
 | 
			
		||||
        err = -EINVAL;
 | 
			
		||||
        goto out;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /* do we need to sync the file? */
 | 
			
		||||
    if (donttouch_stat(&vs->v9stat)) {
 | 
			
		||||
        err = v9fs_do_fsync(s, vs->fidp->fd);
 | 
			
		||||
        v9fs_wstat_post_fsync(s, vs, err);
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (vs->v9stat.mode != -1) {
 | 
			
		||||
        err = v9fs_do_lstat(s, &vs->fidp->path, &vs->stbuf);
 | 
			
		||||
        v9fs_wstat_post_lstat(s, vs, err);
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    v9fs_wstat_post_chmod(s, vs, err);
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
    v9fs_stat_free(&vs->v9stat);
 | 
			
		||||
    complete_pdu(s, vs->pdu, err);
 | 
			
		||||
    qemu_free(vs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
typedef void (pdu_handler_t)(V9fsState *s, V9fsPDU *pdu);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue