/* * 2007+ Copyright (c) Evgeniy Polyakov <zbr@ioremap.net> * All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. */ #include <linux/module.h> #include <linux/backing-dev.h> #include <linux/fs.h> #include <linux/fsnotify.h> #include <linux/mempool.h> #include "netfs.h" static int pohmelfs_send_lock_trans(struct pohmelfs_inode *pi, u64 id, u64 start, u32 size, int type) { struct inode *inode = &pi->vfs_inode; struct pohmelfs_sb *psb = POHMELFS_SB(inode->i_sb); struct netfs_trans *t; struct netfs_cmd *cmd; int path_len, err; void *data; struct netfs_lock *l; int isize = (type & POHMELFS_LOCK_GRAB) ? 0 : sizeof(struct netfs_inode_info); err = pohmelfs_path_length(pi); if (err < 0) goto err_out_exit; path_len = err; err = -ENOMEM; t = netfs_trans_alloc(psb, path_len + sizeof(struct netfs_lock) + isize, NETFS_TRANS_SINGLE_DST, 0); if (!t) goto err_out_exit; cmd = netfs_trans_current(t); data = cmd + 1; err = pohmelfs_construct_path_string(pi, data, path_len); if (err < 0) goto err_out_free; path_len = err; l = data + path_len; l->start = start; l->size = size; l->type = type; l->ino = pi->ino; cmd->cmd = NETFS_LOCK; cmd->start = 0; cmd->id = id; cmd->size = sizeof(struct netfs_lock) + path_len + isize; cmd->ext = path_len; cmd->csize = 0; netfs_convert_cmd(cmd); netfs_convert_lock(l); if (isize) { struct netfs_inode_info *info = (struct netfs_inode_info *)(l + 1); info->mode = inode->i_mode; info->nlink = inode->i_nlink; info->uid = inode->i_uid; info->gid = inode->i_gid; info->blocks = inode->i_blocks; info->rdev = inode->i_rdev; info->size = inode->i_size; info->version = inode->i_version; netfs_convert_inode_info(info); } netfs_trans_update(cmd, t, path_len + sizeof(struct netfs_lock) + isize); return netfs_trans_finish(t, psb); err_out_free: netfs_trans_free(t); err_out_exit: printk("%s: err: %d.\n", __func__, err); return err; } int pohmelfs_data_lock(struct pohmelfs_inode *pi, u64 start, u32 size, int type) { struct pohmelfs_sb *psb = POHMELFS_SB(pi->vfs_inode.i_sb); struct pohmelfs_mcache *m; int err = -ENOMEM; struct iattr iattr; struct inode *inode = &pi->vfs_inode; dprintk("%s: %p: ino: %llu, start: %llu, size: %u, " "type: %d, locked as: %d, owned: %d.\n", __func__, &pi->vfs_inode, pi->ino, start, size, type, pi->lock_type, !!test_bit(NETFS_INODE_OWNED, &pi->state)); if (!pohmelfs_need_lock(pi, type)) return 0; m = pohmelfs_mcache_alloc(psb, start, size, NULL); if (IS_ERR(m)) return PTR_ERR(m); err = pohmelfs_send_lock_trans(pi, m->gen, start, size, type | POHMELFS_LOCK_GRAB); if (err) goto err_out_put; err = wait_for_completion_timeout(&m->complete, psb->mcache_timeout); if (err) err = m->err; else err = -ETIMEDOUT; if (err) { printk("%s: %p: ino: %llu, mgen: %llu, start: %llu, size: %u, err: %d.\n", __func__, &pi->vfs_inode, pi->ino, m->gen, start, size, err); } if (err && (err != -ENOENT)) goto err_out_put; if (!err) { netfs_convert_inode_info(&m->info); iattr.ia_valid = ATTR_MODE | ATTR_UID | ATTR_GID | ATTR_SIZE | ATTR_ATIME; iattr.ia_mode = m->info.mode; iattr.ia_uid = m->info.uid; iattr.ia_gid = m->info.gid; iattr.ia_size = m->info.size; iattr.ia_atime = CURRENT_TIME; dprintk("%s: %p: ino: %llu, mgen: %llu, start: %llu, isize: %llu -> %llu.\n", __func__, &pi->vfs_inode, pi->ino, m->gen, start, inode->i_size, m->info.size); err = pohmelfs_setattr_raw(inode, &iattr); if (!err) { struct dentry *dentry = d_find_alias(inode); if (dentry) { fsnotify_change(dentry, iattr.ia_valid); dput(dentry); } } } pi->lock_type = type; set_bit(NETFS_INODE_OWNED, &pi->state); pohmelfs_mcache_put(psb, m); return 0; err_out_put: pohmelfs_mcache_put(psb, m); return err; } int pohmelfs_data_unlock(struct pohmelfs_inode *pi, u64 start, u32 size, int type) { dprintk("%s: %p: ino: %llu, start: %llu, size: %u, type: %d.\n", __func__, &pi->vfs_inode, pi->ino, start, size, type); pi->lock_type = 0; clear_bit(NETFS_INODE_REMOTE_DIR_SYNCED, &pi->state); clear_bit(NETFS_INODE_OWNED, &pi->state); return pohmelfs_send_lock_trans(pi, pi->ino, start, size, type); }