From: Marco Stornelli <marco.stornelli@xxxxxxxxx> File operations for files. Signed-off-by: Marco Stornelli <marco.stornelli@xxxxxxxxx> --- diff -uprN linux-2.6.30-orig/fs/pramfs/file.c linux-2.6.30/fs/pramfs/file.c --- linux-2.6.30-orig/fs/pramfs/file.c 1970-01-01 01:00:00.000000000 +0100 +++ linux-2.6.30/fs/pramfs/file.c 2009-06-13 12:52:56.000000000 +0200 @@ -0,0 +1,141 @@ +/* + * FILE NAME fs/pramfs/file.c + * + * BRIEF DESCRIPTION + * + * File operations for files. + * + * Copyright 2009 Marco Stornelli <marco.stornelli@xxxxxxxxx> + * Copyright 2003 Sony Corporation + * Copyright 2003 Matsushita Electric Industrial Co., Ltd. + * 2003-2004 (c) MontaVista Software, Inc. , Steve Longerbeam + * This file is licensed under the terms of the GNU General Public + * License version 2. This program is licensed "as is" without any + * warranty of any kind, whether express or implied. + */ +#include <linux/fs.h> +#include <linux/sched.h> +#include <linux/slab.h> +#include <linux/uio.h> +#include <linux/uaccess.h> +#include "pram_fs.h" +#include "xip.h" + +static int pram_open_file(struct inode *inode, struct file *filp) +{ +#ifndef CONFIG_PRAMFS_XIP + /* Without XIP we force to use Direct IO */ + filp->f_flags |= O_DIRECT; +#endif + return generic_file_open(inode, filp); +} + +/* + * Called when an inode is released. Note that this is different + * from pram_open_file: open gets called at every open, but release + * gets called only when /all/ the files are closed. + */ +static int pram_release_file(struct inode *inode, struct file *filp) +{ + return 0; +} + +ssize_t __pram_direct_IO(int rw, struct kiocb *iocb, + const struct iovec *iov, + loff_t offset, unsigned long nr_segs) +{ + struct file *file = iocb->ki_filp; + struct inode *inode = file->f_mapping->host; + struct super_block *sb = inode->i_sb; + int progress = 0; + ssize_t retval = 0; + struct pram_inode *pi; + void *tmp = NULL; + unsigned long flags; + unsigned long blocknr, blockoff; + int num_blocks, blocksize_mask, blocksize, blocksize_bits; + char __user *buf = iov->iov_base; + size_t length = iov->iov_len; + + flags = 0; + + if (length < 0) + return -EINVAL; + if ((rw == READ) && (offset + length > inode->i_size)) + length = inode->i_size - offset; + if (!length) + goto out; + + blocksize_bits = inode->i_sb->s_blocksize_bits; + blocksize = 1 << blocksize_bits; + blocksize_mask = blocksize - 1; + + /* find starting block number to access */ + blocknr = offset >> blocksize_bits; + /* find starting offset within starting block */ + blockoff = offset & blocksize_mask; + /* find number of blocks to access */ + num_blocks = (blockoff + length + blocksize_mask) >> blocksize_bits; + + if (rw == WRITE) { + /* prepare a temporary buffer to hold a user data block + for writing. */ + tmp = kmalloc(blocksize, GFP_KERNEL); + if (!tmp) + return -ENOMEM; + /* now allocate the data blocks we'll need */ + retval = pram_alloc_blocks(inode, blocknr, num_blocks); + if (retval) + goto fail; + } + + pi = pram_get_inode(inode->i_sb, inode->i_ino); + + while (length) { + int count; + off_t block = pram_find_data_block(inode, blocknr++); + u8 *bp = (u8 *)pram_get_block(sb, block); + if (!bp) + goto fail; + + count = blockoff + length > blocksize ? + blocksize - blockoff : length; + + if (rw == READ) { + copy_to_user(buf, &bp[blockoff], count); + } else { + copy_from_user(tmp, buf, count); + + pram_lock_block(inode->i_sb, bp, flags); + memcpy(&bp[blockoff], tmp, count); + pram_unlock_block(inode->i_sb, bp, flags); + } + + progress += count; + buf += count; + length -= count; + blockoff = 0; + } + + retval = progress; + fail: + kfree(tmp); + out: + return retval; +} + +struct file_operations pram_file_operations = { + .llseek = generic_file_llseek, + .read = pram_read, + .write = pram_write, + .aio_read = pram_aio_read, + .aio_write = pram_aio_write, + .mmap = pram_mmap, + .open = pram_open_file, + .release = pram_release_file, + .fsync = simple_sync_file, +}; + +struct inode_operations pram_file_inode_operations = { + .truncate = pram_truncate, +}; -- To unsubscribe from this list: send the line "unsubscribe linux-embedded" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html