OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [rc203soc/] [sw/] [uClinux/] [fs/] [smbfs/] [file.c] - Rev 1771

Go to most recent revision | Compare with Previous | Blame | View Log

/*
 *  file.c
 *
 *  Copyright (C) 1995, 1996 by Paal-Kr. Engstad and Volker Lendecke
 *
 */
 
#include <asm/segment.h>
#include <asm/system.h>
 
#include <linux/sched.h>
#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/fcntl.h>
#include <linux/stat.h>
#include <linux/mm.h>
#include <linux/smb_fs.h>
#include <linux/malloc.h>
 
static inline int
min(int a, int b)
{
	return a < b ? a : b;
}
 
static int
smb_fsync(struct inode *inode, struct file *file)
{
	return 0;
}
 
int
smb_make_open(struct inode *i, int right)
{
	struct smb_dirent *dirent;
 
	if (i == NULL)
	{
		printk("smb_make_open: got NULL inode\n");
		return -EINVAL;
	}
	dirent = &(SMB_INOP(i)->finfo);
 
	DDPRINTK("smb_make_open: dirent->opened = %d\n", dirent->opened);
 
	if ((dirent->opened) == 0)
	{
		/* tries max. rights */
		int open_result = smb_proc_open(SMB_SERVER(i),
						SMB_INOP(i)->dir,
						dirent->name, dirent->len,
						dirent);
		if (open_result)
		{
			return open_result;
		}
	}
	if (((right == O_RDONLY) && ((dirent->access == O_RDONLY)
				     || (dirent->access == O_RDWR)))
	    || ((right == O_WRONLY) && ((dirent->access == O_WRONLY)
					|| (dirent->access == O_RDWR)))
	    || ((right == O_RDWR) && (dirent->access == O_RDWR)))
		return 0;
 
	return -EACCES;
}
 
static int
smb_file_read(struct inode *inode, struct file *file, char *buf, int count)
{
	int result, bufsize, to_read, already_read;
	off_t pos;
	int errno;
 
	DPRINTK("smb_file_read: enter %s\n", SMB_FINFO(inode)->name);
 
	if (!inode)
	{
		DPRINTK("smb_file_read: inode = NULL\n");
		return -EINVAL;
	}
	if (!S_ISREG(inode->i_mode))
	{
		DPRINTK("smb_file_read: read from non-file, mode %07o\n",
			inode->i_mode);
		return -EINVAL;
	}
	if ((errno = smb_make_open(inode, O_RDONLY)) != 0)
		return errno;
 
	pos = file->f_pos;
 
	if (pos + count > inode->i_size)
	{
		count = inode->i_size - pos;
	}
	if (count <= 0)
	{
		return 0;
	}
	bufsize = SMB_SERVER(inode)->max_xmit - SMB_HEADER_LEN - 5 * 2 - 5;
 
	already_read = 0;
 
	/* First read in as much as possible for each bufsize. */
	while (already_read < count)
	{
		to_read = min(bufsize, count - already_read);
		result = smb_proc_read(SMB_SERVER(inode), SMB_FINFO(inode),
				       pos, to_read, buf, 1);
		if (result < 0)
		{
			return result;
		}
		pos += result;
		buf += result;
		already_read += result;
 
		if (result < to_read)
		{
			break;
		}
	}
 
	file->f_pos = pos;
 
	if (!IS_RDONLY(inode))
		inode->i_atime = CURRENT_TIME;
	inode->i_dirt = 1;
 
	DPRINTK("smb_file_read: exit %s\n", SMB_FINFO(inode)->name);
 
	return already_read;
}
 
static int
smb_file_write(struct inode *inode, struct file *file, const char *buf,
	       int count)
{
	int result, bufsize, to_write, already_written;
	off_t pos;
	int errno;
 
	if (!inode)
	{
		DPRINTK("smb_file_write: inode = NULL\n");
		return -EINVAL;
	}
	if (!S_ISREG(inode->i_mode))
	{
		DPRINTK("smb_file_write: write to non-file, mode %07o\n",
			inode->i_mode);
		return -EINVAL;
	}
	DPRINTK("smb_file_write: enter %s\n", SMB_FINFO(inode)->name);
 
	if (count <= 0)
	{
		return 0;
	}
	if ((errno = smb_make_open(inode, O_RDWR)) != 0)
	{
		return errno;
	}
	pos = file->f_pos;
 
	if (file->f_flags & O_APPEND)
		pos = inode->i_size;
 
	bufsize = SMB_SERVER(inode)->max_xmit - SMB_HEADER_LEN - 5 * 2 - 5;
 
	already_written = 0;
 
	DPRINTK("smb_write_file: blkmode = %d, blkmode & 2 = %d\n",
		SMB_SERVER(inode)->blkmode,
		SMB_SERVER(inode)->blkmode & 2);
 
	while (already_written < count)
	{
		to_write = min(bufsize, count - already_written);
		result = smb_proc_write(SMB_SERVER(inode), SMB_FINFO(inode),
					pos, to_write, buf);
 
		if (result < 0)
		{
			return result;
		}
		pos += result;
		buf += result;
		already_written += result;
 
		if (result < to_write)
		{
			break;
		}
	}
 
	inode->i_mtime = inode->i_ctime = CURRENT_TIME;
	inode->i_dirt = 1;
 
	file->f_pos = pos;
 
	if (pos > inode->i_size)
	{
		inode->i_size = pos;
	}
	DPRINTK("smb_file_write: exit %s\n", SMB_FINFO(inode)->name);
 
	return already_written;
}
 
static struct file_operations smb_file_operations =
{
	NULL,			/* lseek - default */
	smb_file_read,		/* read */
	smb_file_write,		/* write */
	NULL,			/* readdir - bad */
	NULL,			/* select - default */
	smb_ioctl,		/* ioctl */
	smb_mmap,		/* mmap */
	NULL,			/* open */
	NULL,			/* release */
	smb_fsync,		/* fsync */
};
 
struct inode_operations smb_file_inode_operations =
{
	&smb_file_operations,	/* default file operations */
	NULL,			/* create */
	NULL,			/* lookup */
	NULL,			/* link */
	NULL,			/* unlink */
	NULL,			/* symlink */
	NULL,			/* mkdir */
	NULL,			/* rmdir */
	NULL,			/* mknod */
	NULL,			/* rename */
	NULL,			/* readlink */
	NULL,			/* follow_link */
	NULL,			/* readpage */
	NULL,			/* writepage */
	NULL,			/* bmap */
	NULL			/* truncate */
};
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.