⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 dir.c

📁 嵌入式系统设计与实例开发实验教材二源码 多线程应用程序设计 串行端口程序设计 AD接口实验 CAN总线通信实验 GPS通信实验 Linux内核移植与编译实验 IC卡读写实验 SD驱动使
💻 C
字号:
/* *  dir.c * *  Copyright (C) 1995, 1996 by Paal-Kr. Engstad and Volker Lendecke *  Copyright (C) 1997 by Volker Lendecke * *  Please add a note about your changes to smbfs in the ChangeLog file. */#include <linux/sched.h>#include <linux/errno.h>#include <linux/kernel.h>#include <linux/smp_lock.h>#include <linux/ctype.h>#include <linux/smb_fs.h>#include <linux/smb_mount.h>#include <linux/smbno.h>#include "smb_debug.h"#include "proto.h"static int smb_readdir(struct file *, void *, filldir_t);static int smb_dir_open(struct inode *, struct file *);static struct dentry *smb_lookup(struct inode *, struct dentry *);static int smb_create(struct inode *, struct dentry *, int);static int smb_mkdir(struct inode *, struct dentry *, int);static int smb_rmdir(struct inode *, struct dentry *);static int smb_unlink(struct inode *, struct dentry *);static int smb_rename(struct inode *, struct dentry *,		      struct inode *, struct dentry *);struct file_operations smb_dir_operations ={	read:		generic_read_dir,	readdir:	smb_readdir,	ioctl:		smb_ioctl,	open:		smb_dir_open,};struct inode_operations smb_dir_inode_operations ={	create:		smb_create,	lookup:		smb_lookup,	unlink:		smb_unlink,	mkdir:		smb_mkdir,	rmdir:		smb_rmdir,	rename:		smb_rename,	revalidate:	smb_revalidate_inode,	setattr:	smb_notify_change,};/* * Read a directory, using filldir to fill the dirent memory. * smb_proc_readdir does the actual reading from the smb server. * * The cache code is almost directly taken from ncpfs */static int smb_readdir(struct file *filp, void *dirent, filldir_t filldir){	struct dentry *dentry = filp->f_dentry;	struct inode *dir = dentry->d_inode;	struct smb_sb_info *server = server_from_dentry(dentry);	union  smb_dir_cache *cache = NULL;	struct smb_cache_control ctl;	struct page *page = NULL;	int result;	ctl.page  = NULL;	ctl.cache = NULL;	VERBOSE("reading %s/%s, f_pos=%d\n",		DENTRY_PATH(dentry),  (int) filp->f_pos);	result = 0;	switch ((unsigned int) filp->f_pos) {	case 0:		if (filldir(dirent, ".", 1, 0, dir->i_ino, DT_DIR) < 0)			goto out;		filp->f_pos = 1;		/* fallthrough */	case 1:		if (filldir(dirent, "..", 2, 1,			    dentry->d_parent->d_inode->i_ino, DT_DIR) < 0)			goto out;		filp->f_pos = 2;	}	/*	 * Make sure our inode is up-to-date.	 */	result = smb_revalidate_inode(dentry);	if (result)		goto out;	page = grab_cache_page(&dir->i_data, 0);	if (!page)		goto read_really;	ctl.cache = cache = kmap(page);	ctl.head  = cache->head;	if (!Page_Uptodate(page) || !ctl.head.eof) {		VERBOSE("%s/%s, page uptodate=%d, eof=%d\n",			 DENTRY_PATH(dentry), Page_Uptodate(page),ctl.head.eof);		goto init_cache;	}	if (filp->f_pos == 2) {		if (jiffies - ctl.head.time >= SMB_MAX_AGE(server))			goto init_cache;		/*		 * N.B. ncpfs checks mtime of dentry too here, we don't.		 *   1. common smb servers do not update mtime on dir changes		 *   2. it requires an extra smb request		 *      (revalidate has the same timeout as ctl.head.time)		 *		 * Instead smbfs invalidates its own cache on local changes		 * and remote changes are not seen until timeout.		 */	}	if (filp->f_pos > ctl.head.end)		goto finished;	ctl.fpos = filp->f_pos + (SMB_DIRCACHE_START - 2);	ctl.ofs  = ctl.fpos / SMB_DIRCACHE_SIZE;	ctl.idx  = ctl.fpos % SMB_DIRCACHE_SIZE;	for (;;) {		if (ctl.ofs != 0) {			ctl.page = find_lock_page(&dir->i_data, ctl.ofs);			if (!ctl.page)				goto invalid_cache;			ctl.cache = kmap(ctl.page);			if (!Page_Uptodate(ctl.page))				goto invalid_cache;		}		while (ctl.idx < SMB_DIRCACHE_SIZE) {			struct dentry *dent;			int res;			dent = smb_dget_fpos(ctl.cache->dentry[ctl.idx],					     dentry, filp->f_pos);			if (!dent)				goto invalid_cache;			res = filldir(dirent, dent->d_name.name,				      dent->d_name.len, filp->f_pos,				      dent->d_inode->i_ino, DT_UNKNOWN);			dput(dent);			if (res)				goto finished;			filp->f_pos += 1;			ctl.idx += 1;			if (filp->f_pos > ctl.head.end)				goto finished;		}		if (ctl.page) {			kunmap(ctl.page);			SetPageUptodate(ctl.page);			UnlockPage(ctl.page);			page_cache_release(ctl.page);			ctl.page = NULL;		}		ctl.idx  = 0;		ctl.ofs += 1;	}invalid_cache:	if (ctl.page) {		kunmap(ctl.page);		UnlockPage(ctl.page);		page_cache_release(ctl.page);		ctl.page = NULL;	}	ctl.cache = cache;init_cache:	smb_invalidate_dircache_entries(dentry);	ctl.head.time = jiffies;	ctl.head.eof = 0;	ctl.fpos = 2;	ctl.ofs = 0;	ctl.idx = SMB_DIRCACHE_START;	ctl.filled = 0;	ctl.valid  = 1;read_really:	result = smb_proc_readdir(filp, dirent, filldir, &ctl);	if (ctl.idx == -1)		goto invalid_cache;	/* retry */	ctl.head.end = ctl.fpos - 1;	ctl.head.eof = ctl.valid;finished:	if (page) {		cache->head = ctl.head;		kunmap(page);		SetPageUptodate(page);		UnlockPage(page);		page_cache_release(page);	}	if (ctl.page) {		kunmap(ctl.page);		SetPageUptodate(ctl.page);		UnlockPage(ctl.page);		page_cache_release(ctl.page);	}out:	return result;}static intsmb_dir_open(struct inode *dir, struct file *file){	struct dentry *dentry = file->f_dentry;	struct smb_sb_info *server;	int error = 0;	VERBOSE("(%s/%s)\n", dentry->d_parent->d_name.name,		file->f_dentry->d_name.name);	/*	 * Directory timestamps in the core protocol aren't updated	 * when a file is added, so we give them a very short TTL.	 */	lock_kernel();	server = server_from_dentry(dentry);	if (server->opt.protocol < SMB_PROTOCOL_LANMAN2) {		unsigned long age = jiffies - dir->u.smbfs_i.oldmtime;		if (age > 2*HZ)			smb_invalid_dir_cache(dir);	}	/*	 * Note: in order to allow the smbmount process to open the	 * mount point, we only revalidate if the connection is valid or	 * if the process is trying to access something other than the root.	 */	if (server->state == CONN_VALID || !IS_ROOT(dentry))		error = smb_revalidate_inode(dentry);	unlock_kernel();	return error;}/* * Dentry operations routines */static int smb_lookup_validate(struct dentry *, int);static int smb_hash_dentry(struct dentry *, struct qstr *);static int smb_compare_dentry(struct dentry *, struct qstr *, struct qstr *);static int smb_delete_dentry(struct dentry *);static struct dentry_operations smbfs_dentry_operations ={	d_revalidate:	smb_lookup_validate,	d_hash:		smb_hash_dentry,	d_compare:	smb_compare_dentry,	d_delete:	smb_delete_dentry,};static struct dentry_operations smbfs_dentry_operations_case ={	d_revalidate:	smb_lookup_validate,	d_delete:	smb_delete_dentry,};/* * This is the callback when the dcache has a lookup hit. */static intsmb_lookup_validate(struct dentry * dentry, int flags){	struct smb_sb_info *server = server_from_dentry(dentry);	struct inode * inode = dentry->d_inode;	unsigned long age = jiffies - dentry->d_time;	int valid;	/*	 * The default validation is based on dentry age:	 * we believe in dentries for a few seconds.  (But each	 * successful server lookup renews the timestamp.)	 */	valid = (age <= SMB_MAX_AGE(server));#ifdef SMBFS_DEBUG_VERBOSE	if (!valid)		VERBOSE("%s/%s not valid, age=%lu\n", 			DENTRY_PATH(dentry), age);#endif	if (inode) {		lock_kernel();		if (is_bad_inode(inode)) {			PARANOIA("%s/%s has dud inode\n", DENTRY_PATH(dentry));			valid = 0;		} else if (!valid)			valid = (smb_revalidate_inode(dentry) == 0);		unlock_kernel();	} else {		/*		 * What should we do for negative dentries?		 */	}	return valid;}static int smb_hash_dentry(struct dentry *dir, struct qstr *this){	unsigned long hash;	int i;	hash = init_name_hash();	for (i=0; i < this->len ; i++)		hash = partial_name_hash(tolower(this->name[i]), hash);	this->hash = end_name_hash(hash);  	return 0;}static intsmb_compare_dentry(struct dentry *dir, struct qstr *a, struct qstr *b){	int i, result = 1;	if (a->len != b->len)		goto out;	for (i=0; i < a->len; i++) {		if (tolower(a->name[i]) != tolower(b->name[i]))			goto out;	}	result = 0;out:	return result;}/* * This is the callback from dput() when d_count is going to 0. * We use this to unhash dentries with bad inodes. */static intsmb_delete_dentry(struct dentry * dentry){	if (dentry->d_inode) {		if (is_bad_inode(dentry->d_inode)) {			PARANOIA("bad inode, unhashing %s/%s\n",				 DENTRY_PATH(dentry));			return 1;		}	} else {		/* N.B. Unhash negative dentries? */	}	return 0;}/* * Initialize a new dentry */voidsmb_new_dentry(struct dentry *dentry){	struct smb_sb_info *server = server_from_dentry(dentry);	if (server->mnt->flags & SMB_MOUNT_CASE)		dentry->d_op = &smbfs_dentry_operations_case;	else		dentry->d_op = &smbfs_dentry_operations;	dentry->d_time = jiffies;}/* * Whenever a lookup succeeds, we know the parent directories * are all valid, so we want to update the dentry timestamps. * N.B. Move this to dcache? */voidsmb_renew_times(struct dentry * dentry){	for (;;) {		dentry->d_time = jiffies;		if (IS_ROOT(dentry))			break;		dentry = dentry->d_parent;	}}static struct dentry *smb_lookup(struct inode *dir, struct dentry *dentry){	struct smb_fattr finfo;	struct inode *inode;	int error;	struct smb_sb_info *server;	error = -ENAMETOOLONG;	if (dentry->d_name.len > SMB_MAXNAMELEN)		goto out;	error = smb_proc_getattr(dentry, &finfo);#ifdef SMBFS_PARANOIA	if (error && error != -ENOENT)		PARANOIA("find %s/%s failed, error=%d\n",			 DENTRY_PATH(dentry), error);#endif	inode = NULL;	if (error == -ENOENT)		goto add_entry;	if (!error) {		error = -EACCES;		finfo.f_ino = iunique(dentry->d_sb, 2);		inode = smb_iget(dir->i_sb, &finfo);		if (inode) {	add_entry:			server = server_from_dentry(dentry);			if (server->mnt->flags & SMB_MOUNT_CASE)				dentry->d_op = &smbfs_dentry_operations_case;			else				dentry->d_op = &smbfs_dentry_operations;			d_add(dentry, inode);			smb_renew_times(dentry);			error = 0;		}	}out:	return ERR_PTR(error);}/* * This code is common to all routines creating a new inode. */static intsmb_instantiate(struct dentry *dentry, __u16 fileid, int have_id){	struct smb_sb_info *server = server_from_dentry(dentry);	struct inode *inode;	int error;	struct smb_fattr fattr;	VERBOSE("file %s/%s, fileid=%u\n", DENTRY_PATH(dentry), fileid);	error = smb_proc_getattr(dentry, &fattr);	if (error)		goto out_close;	smb_renew_times(dentry);	fattr.f_ino = iunique(dentry->d_sb, 2);	inode = smb_iget(dentry->d_sb, &fattr);	if (!inode)		goto out_no_inode;	if (have_id) {		inode->u.smbfs_i.fileid = fileid;		inode->u.smbfs_i.access = SMB_O_RDWR;		inode->u.smbfs_i.open = server->generation;	}	d_instantiate(dentry, inode);out:	return error;out_no_inode:	error = -EACCES;out_close:	if (have_id) {		PARANOIA("%s/%s failed, error=%d, closing %u\n",			 DENTRY_PATH(dentry), error, fileid);		smb_close_fileid(dentry, fileid);	}	goto out;}/* N.B. How should the mode argument be used? */static intsmb_create(struct inode *dir, struct dentry *dentry, int mode){	__u16 fileid;	int error;	VERBOSE("creating %s/%s, mode=%d\n", DENTRY_PATH(dentry), mode);	smb_invalid_dir_cache(dir);	error = smb_proc_create(dentry, 0, CURRENT_TIME, &fileid);	if (!error) {		error = smb_instantiate(dentry, fileid, 1);	} else {		PARANOIA("%s/%s failed, error=%d\n",			 DENTRY_PATH(dentry), error);	}	return error;}/* N.B. How should the mode argument be used? */static intsmb_mkdir(struct inode *dir, struct dentry *dentry, int mode){	int error;	smb_invalid_dir_cache(dir);	error = smb_proc_mkdir(dentry);	if (!error) {		error = smb_instantiate(dentry, 0, 0);	}	return error;}static intsmb_rmdir(struct inode *dir, struct dentry *dentry){	struct inode *inode = dentry->d_inode;	int error;	/*	 * Close the directory if it's open.	 */	smb_close(inode);	/*	 * Check that nobody else is using the directory..	 */	error = -EBUSY;	if (!d_unhashed(dentry))		goto out;	smb_invalid_dir_cache(dir);	error = smb_proc_rmdir(dentry);out:	return error;}static intsmb_unlink(struct inode *dir, struct dentry *dentry){	int error;	/*	 * Close the file if it's open.	 */	smb_close(dentry->d_inode);	smb_invalid_dir_cache(dir);	error = smb_proc_unlink(dentry);	if (!error)		smb_renew_times(dentry);	return error;}static intsmb_rename(struct inode *old_dir, struct dentry *old_dentry,	   struct inode *new_dir, struct dentry *new_dentry){	int error;	/*	 * Close any open files, and check whether to delete the	 * target before attempting the rename.	 */	if (old_dentry->d_inode)		smb_close(old_dentry->d_inode);	if (new_dentry->d_inode) {		smb_close(new_dentry->d_inode);		error = smb_proc_unlink(new_dentry);		if (error) {			VERBOSE("unlink %s/%s, error=%d\n",				DENTRY_PATH(new_dentry), error);			goto out;		}		/* FIXME */		d_delete(new_dentry);	}	smb_invalid_dir_cache(old_dir);	smb_invalid_dir_cache(new_dir);	error = smb_proc_mv(old_dentry, new_dentry);	if (!error) {		smb_renew_times(old_dentry);		smb_renew_times(new_dentry);	}out:	return error;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -