fs: handle SEEK_HOLE/SEEK_DATA properly in all fs's that define their own llseek
This converts everybody to handle SEEK_HOLE/SEEK_DATA properly. In some cases we just return -EINVAL, in others we do the normal generic thing, and in others we're simply making sure that the properly due-dilligence is done. For example in NFS/CIFS we need to make sure the file size is update properly for the SEEK_HOLE and SEEK_DATA case, but since it calls the generic llseek stuff itself that is all we have to do. Thanks, Signed-off-by: Josef Bacik <josef@redhat.com> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
		
					parent
					
						
							
								c334b1138b
							
						
					
				
			
			
				commit
				
					
						06222e491e
					
				
			
		
					 7 changed files with 66 additions and 12 deletions
				
			
		| 
						 | 
				
			
			@ -355,20 +355,25 @@ static loff_t block_llseek(struct file *file, loff_t offset, int origin)
 | 
			
		|||
	mutex_lock(&bd_inode->i_mutex);
 | 
			
		||||
	size = i_size_read(bd_inode);
 | 
			
		||||
 | 
			
		||||
	retval = -EINVAL;
 | 
			
		||||
	switch (origin) {
 | 
			
		||||
		case 2:
 | 
			
		||||
		case SEEK_END:
 | 
			
		||||
			offset += size;
 | 
			
		||||
			break;
 | 
			
		||||
		case 1:
 | 
			
		||||
		case SEEK_CUR:
 | 
			
		||||
			offset += file->f_pos;
 | 
			
		||||
		case SEEK_SET:
 | 
			
		||||
			break;
 | 
			
		||||
		default:
 | 
			
		||||
			goto out;
 | 
			
		||||
	}
 | 
			
		||||
	retval = -EINVAL;
 | 
			
		||||
	if (offset >= 0 && offset <= size) {
 | 
			
		||||
		if (offset != file->f_pos) {
 | 
			
		||||
			file->f_pos = offset;
 | 
			
		||||
		}
 | 
			
		||||
		retval = offset;
 | 
			
		||||
	}
 | 
			
		||||
out:
 | 
			
		||||
	mutex_unlock(&bd_inode->i_mutex);
 | 
			
		||||
	return retval;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -446,14 +446,19 @@ static loff_t ceph_dir_llseek(struct file *file, loff_t offset, int origin)
 | 
			
		|||
	loff_t retval;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&inode->i_mutex);
 | 
			
		||||
	retval = -EINVAL;
 | 
			
		||||
	switch (origin) {
 | 
			
		||||
	case SEEK_END:
 | 
			
		||||
		offset += inode->i_size + 2;   /* FIXME */
 | 
			
		||||
		break;
 | 
			
		||||
	case SEEK_CUR:
 | 
			
		||||
		offset += file->f_pos;
 | 
			
		||||
	case SEEK_SET:
 | 
			
		||||
		break;
 | 
			
		||||
	default:
 | 
			
		||||
		goto out;
 | 
			
		||||
	}
 | 
			
		||||
	retval = -EINVAL;
 | 
			
		||||
 | 
			
		||||
	if (offset >= 0 && offset <= inode->i_sb->s_maxbytes) {
 | 
			
		||||
		if (offset != file->f_pos) {
 | 
			
		||||
			file->f_pos = offset;
 | 
			
		||||
| 
						 | 
				
			
			@ -477,6 +482,7 @@ static loff_t ceph_dir_llseek(struct file *file, loff_t offset, int origin)
 | 
			
		|||
		if (offset > old_offset)
 | 
			
		||||
			fi->dir_release_count--;
 | 
			
		||||
	}
 | 
			
		||||
out:
 | 
			
		||||
	mutex_unlock(&inode->i_mutex);
 | 
			
		||||
	return retval;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -768,13 +768,16 @@ static loff_t ceph_llseek(struct file *file, loff_t offset, int origin)
 | 
			
		|||
 | 
			
		||||
	mutex_lock(&inode->i_mutex);
 | 
			
		||||
	__ceph_do_pending_vmtruncate(inode);
 | 
			
		||||
	switch (origin) {
 | 
			
		||||
	case SEEK_END:
 | 
			
		||||
	if (origin != SEEK_CUR || origin != SEEK_SET) {
 | 
			
		||||
		ret = ceph_do_getattr(inode, CEPH_STAT_CAP_SIZE);
 | 
			
		||||
		if (ret < 0) {
 | 
			
		||||
			offset = ret;
 | 
			
		||||
			goto out;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	switch (origin) {
 | 
			
		||||
	case SEEK_END:
 | 
			
		||||
		offset += inode->i_size;
 | 
			
		||||
		break;
 | 
			
		||||
	case SEEK_CUR:
 | 
			
		||||
| 
						 | 
				
			
			@ -790,6 +793,19 @@ static loff_t ceph_llseek(struct file *file, loff_t offset, int origin)
 | 
			
		|||
		}
 | 
			
		||||
		offset += file->f_pos;
 | 
			
		||||
		break;
 | 
			
		||||
	case SEEK_DATA:
 | 
			
		||||
		if (offset >= inode->i_size) {
 | 
			
		||||
			ret = -ENXIO;
 | 
			
		||||
			goto out;
 | 
			
		||||
		}
 | 
			
		||||
		break;
 | 
			
		||||
	case SEEK_HOLE:
 | 
			
		||||
		if (offset >= inode->i_size) {
 | 
			
		||||
			ret = -ENXIO;
 | 
			
		||||
			goto out;
 | 
			
		||||
		}
 | 
			
		||||
		offset = inode->i_size;
 | 
			
		||||
		break;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (offset < 0 || offset > inode->i_sb->s_maxbytes) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -704,8 +704,11 @@ static ssize_t cifs_file_aio_write(struct kiocb *iocb, const struct iovec *iov,
 | 
			
		|||
 | 
			
		||||
static loff_t cifs_llseek(struct file *file, loff_t offset, int origin)
 | 
			
		||||
{
 | 
			
		||||
	/* origin == SEEK_END => we must revalidate the cached file length */
 | 
			
		||||
	if (origin == SEEK_END) {
 | 
			
		||||
	/*
 | 
			
		||||
	 * origin == SEEK_END || SEEK_DATA || SEEK_HOLE => we must revalidate
 | 
			
		||||
	 * the cached file length
 | 
			
		||||
	 */
 | 
			
		||||
	if (origin != SEEK_SET || origin != SEEK_CUR) {
 | 
			
		||||
		int rc;
 | 
			
		||||
		struct inode *inode = file->f_path.dentry->d_inode;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1600,15 +1600,32 @@ static loff_t fuse_file_llseek(struct file *file, loff_t offset, int origin)
 | 
			
		|||
	struct inode *inode = file->f_path.dentry->d_inode;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&inode->i_mutex);
 | 
			
		||||
	switch (origin) {
 | 
			
		||||
	case SEEK_END:
 | 
			
		||||
	if (origin != SEEK_CUR || origin != SEEK_SET) {
 | 
			
		||||
		retval = fuse_update_attributes(inode, NULL, file, NULL);
 | 
			
		||||
		if (retval)
 | 
			
		||||
			goto exit;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	switch (origin) {
 | 
			
		||||
	case SEEK_END:
 | 
			
		||||
		offset += i_size_read(inode);
 | 
			
		||||
		break;
 | 
			
		||||
	case SEEK_CUR:
 | 
			
		||||
		offset += file->f_pos;
 | 
			
		||||
		break;
 | 
			
		||||
	case SEEK_DATA:
 | 
			
		||||
		if (offset >= i_size_read(inode)) {
 | 
			
		||||
			retval = -ENXIO;
 | 
			
		||||
			goto exit;
 | 
			
		||||
		}
 | 
			
		||||
		break;
 | 
			
		||||
	case SEEK_HOLE:
 | 
			
		||||
		if (offset >= i_size_read(inode)) {
 | 
			
		||||
			retval = -ENXIO;
 | 
			
		||||
			goto exit;
 | 
			
		||||
		}
 | 
			
		||||
		offset = i_size_read(inode);
 | 
			
		||||
		break;
 | 
			
		||||
	}
 | 
			
		||||
	retval = -EINVAL;
 | 
			
		||||
	if (offset >= 0 && offset <= inode->i_sb->s_maxbytes) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -29,6 +29,10 @@ static loff_t hpfs_dir_lseek(struct file *filp, loff_t off, int whence)
 | 
			
		|||
	struct hpfs_inode_info *hpfs_inode = hpfs_i(i);
 | 
			
		||||
	struct super_block *s = i->i_sb;
 | 
			
		||||
 | 
			
		||||
	/* Somebody else will have to figure out what to do here */
 | 
			
		||||
	if (whence == SEEK_DATA || whence == SEEK_HOLE)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	hpfs_lock(s);
 | 
			
		||||
 | 
			
		||||
	/*printk("dir lseek\n");*/
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -187,8 +187,11 @@ static loff_t nfs_file_llseek(struct file *filp, loff_t offset, int origin)
 | 
			
		|||
			filp->f_path.dentry->d_name.name,
 | 
			
		||||
			offset, origin);
 | 
			
		||||
 | 
			
		||||
	/* origin == SEEK_END => we must revalidate the cached file length */
 | 
			
		||||
	if (origin == SEEK_END) {
 | 
			
		||||
	/*
 | 
			
		||||
	 * origin == SEEK_END || SEEK_DATA || SEEK_HOLE => we must revalidate
 | 
			
		||||
	 * the cached file length
 | 
			
		||||
	 */
 | 
			
		||||
	if (origin != SEEK_SET || origin != SEEK_CUR) {
 | 
			
		||||
		struct inode *inode = filp->f_mapping->host;
 | 
			
		||||
 | 
			
		||||
		int retval = nfs_revalidate_file_size(inode, filp);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue