diff options
Diffstat (limited to 'fs/cifs')
| -rw-r--r-- | fs/cifs/cifsglob.h | 4 | ||||
| -rw-r--r-- | fs/cifs/cifsproto.h | 1 | ||||
| -rw-r--r-- | fs/cifs/connect.c | 24 | ||||
| -rw-r--r-- | fs/cifs/dir.c | 6 | ||||
| -rw-r--r-- | fs/cifs/inode.c | 7 | ||||
| -rw-r--r-- | fs/cifs/misc.c | 14 | ||||
| -rw-r--r-- | fs/cifs/readdir.c | 8 | 
7 files changed, 37 insertions, 27 deletions
| diff --git a/fs/cifs/cifsglob.h b/fs/cifs/cifsglob.h index 6084d6379c03..3cfec696af9c 100644 --- a/fs/cifs/cifsglob.h +++ b/fs/cifs/cifsglob.h @@ -572,9 +572,9 @@ require use of the stronger protocol */  #define   CIFSSEC_MUST_LANMAN	0x10010  #define   CIFSSEC_MUST_PLNTXT	0x20020  #ifdef CONFIG_CIFS_UPCALL -#define   CIFSSEC_MASK          0xAF0AF /* allows weak security but also krb5 */ +#define   CIFSSEC_MASK          0xBF0BF /* allows weak security but also krb5 */  #else -#define   CIFSSEC_MASK          0xA70A7 /* current flags supported if weak */ +#define   CIFSSEC_MASK          0xB70B7 /* current flags supported if weak */  #endif /* UPCALL */  #else /* do not allow weak pw hash */  #ifdef CONFIG_CIFS_UPCALL diff --git a/fs/cifs/cifsproto.h b/fs/cifs/cifsproto.h index da8fbf565991..3949ae197809 100644 --- a/fs/cifs/cifsproto.h +++ b/fs/cifs/cifsproto.h @@ -389,4 +389,5 @@ extern int CIFSSMBSetPosixACL(const int xid, struct cifsTconInfo *tcon,  		const struct nls_table *nls_codepage, int remap_special_chars);  extern int CIFSGetExtAttr(const int xid, struct cifsTconInfo *tcon,  			const int netfid, __u64 *pExtAttrBits, __u64 *pMask); +extern void cifs_autodisable_serverino(struct cifs_sb_info *cifs_sb);  #endif			/* _CIFSPROTO_H */ diff --git a/fs/cifs/connect.c b/fs/cifs/connect.c index 1f3345d7fa79..08a44ca2a8db 100644 --- a/fs/cifs/connect.c +++ b/fs/cifs/connect.c @@ -1556,7 +1556,8 @@ cifs_get_tcp_session(struct smb_vol *volume_info)  out_err:  	if (tcp_ses) { -		kfree(tcp_ses->hostname); +		if (!IS_ERR(tcp_ses->hostname)) +			kfree(tcp_ses->hostname);  		if (tcp_ses->ssocket)  			sock_release(tcp_ses->ssocket);  		kfree(tcp_ses); @@ -2199,16 +2200,8 @@ is_path_accessible(int xid, struct cifsTconInfo *tcon,  		   struct cifs_sb_info *cifs_sb, const char *full_path)  {  	int rc; -	__u64 inode_num;  	FILE_ALL_INFO *pfile_info; -	rc = CIFSGetSrvInodeNumber(xid, tcon, full_path, &inode_num, -				   cifs_sb->local_nls, -				   cifs_sb->mnt_cifs_flags & -						CIFS_MOUNT_MAP_SPECIAL_CHR); -	if (rc != -EOPNOTSUPP) -		return rc; -  	pfile_info = kmalloc(sizeof(FILE_ALL_INFO), GFP_KERNEL);  	if (pfile_info == NULL)  		return -ENOMEM; @@ -2274,12 +2267,12 @@ int  cifs_mount(struct super_block *sb, struct cifs_sb_info *cifs_sb,  		char *mount_data_global, const char *devname)  { -	int rc = 0; +	int rc;  	int xid;  	struct smb_vol *volume_info; -	struct cifsSesInfo *pSesInfo = NULL; -	struct cifsTconInfo *tcon = NULL; -	struct TCP_Server_Info *srvTcp = NULL; +	struct cifsSesInfo *pSesInfo; +	struct cifsTconInfo *tcon; +	struct TCP_Server_Info *srvTcp;  	char   *full_path;  	char *mount_data = mount_data_global;  #ifdef CONFIG_CIFS_DFS_UPCALL @@ -2288,6 +2281,10 @@ cifs_mount(struct super_block *sb, struct cifs_sb_info *cifs_sb,  	int referral_walks_count = 0;  try_mount_again:  #endif +	rc = 0; +	tcon = NULL; +	pSesInfo = NULL; +	srvTcp = NULL;  	full_path = NULL;  	xid = GetXid(); @@ -2584,6 +2581,7 @@ remote_path_check:  			cleanup_volume_info(&volume_info);  			referral_walks_count++; +			FreeXid(xid);  			goto try_mount_again;  		}  #else /* No DFS support, return error on mount */ diff --git a/fs/cifs/dir.c b/fs/cifs/dir.c index 4326ffd90fa9..965269041c31 100644 --- a/fs/cifs/dir.c +++ b/fs/cifs/dir.c @@ -212,8 +212,6 @@ int cifs_posix_open(char *full_path, struct inode **pinode,  		posix_flags |= SMB_O_EXCL;  	if (oflags & O_TRUNC)  		posix_flags |= SMB_O_TRUNC; -	if (oflags & O_APPEND) -		posix_flags |= SMB_O_APPEND;  	if (oflags & O_SYNC)  		posix_flags |= SMB_O_SYNC;  	if (oflags & O_DIRECTORY) @@ -648,7 +646,7 @@ cifs_lookup(struct inode *parent_dir_inode, struct dentry *direntry,  	 * O_EXCL: optimize away the lookup, but don't hash the dentry. Let  	 * the VFS handle the create.  	 */ -	if (nd->flags & LOOKUP_EXCL) { +	if (nd && (nd->flags & LOOKUP_EXCL)) {  		d_instantiate(direntry, NULL);  		return 0;  	} @@ -680,7 +678,7 @@ cifs_lookup(struct inode *parent_dir_inode, struct dentry *direntry,  	 * reduction in network traffic in the other paths.  	 */  	if (pTcon->unix_ext) { -		if (!(nd->flags & (LOOKUP_PARENT | LOOKUP_DIRECTORY)) && +		if (nd && !(nd->flags & (LOOKUP_PARENT | LOOKUP_DIRECTORY)) &&  		     (nd->flags & LOOKUP_OPEN) && !pTcon->broken_posix_open &&  		     (nd->intent.open.flags & O_CREAT)) {  			rc = cifs_posix_open(full_path, &newInode, diff --git a/fs/cifs/inode.c b/fs/cifs/inode.c index 82d83839655e..9a77a30756c4 100644 --- a/fs/cifs/inode.c +++ b/fs/cifs/inode.c @@ -512,13 +512,10 @@ int cifs_get_inode_info(struct inode **pinode,  					cifs_sb->local_nls,  					cifs_sb->mnt_cifs_flags &  						CIFS_MOUNT_MAP_SPECIAL_CHR); -			if (rc1) { +			if (rc1 || !fattr.cf_uniqueid) {  				cFYI(1, ("GetSrvInodeNum rc %d", rc1));  				fattr.cf_uniqueid = iunique(sb, ROOT_I); -				/* disable serverino if call not supported */ -				if (rc1 == -EINVAL) -					cifs_sb->mnt_cifs_flags &= -							~CIFS_MOUNT_SERVER_INUM; +				cifs_autodisable_serverino(cifs_sb);  			}  		} else {  			fattr.cf_uniqueid = iunique(sb, ROOT_I); diff --git a/fs/cifs/misc.c b/fs/cifs/misc.c index e079a9190ec4..b8dc3ed5d46c 100644 --- a/fs/cifs/misc.c +++ b/fs/cifs/misc.c @@ -705,3 +705,17 @@ cifsConvertToUCS(__le16 *target, const char *source, int maxlen,  ctoUCS_out:  	return i;  } + +void +cifs_autodisable_serverino(struct cifs_sb_info *cifs_sb) +{ +	if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) { +		cifs_sb->mnt_cifs_flags &= ~CIFS_MOUNT_SERVER_INUM; +		cERROR(1, ("Autodisabling the use of server inode numbers on " +			   "%s. This server doesn't seem to support them " +			   "properly. Hardlinks will not be recognized on this " +			   "mount. Consider mounting with the \"noserverino\" " +			   "option to silence this message.", +			   cifs_sb->tcon->treeName)); +	} +} diff --git a/fs/cifs/readdir.c b/fs/cifs/readdir.c index f823a4a208a7..f55d42bbf7e8 100644 --- a/fs/cifs/readdir.c +++ b/fs/cifs/readdir.c @@ -666,6 +666,7 @@ static int cifs_get_name_from_search_buf(struct qstr *pqst,  					   min(len, max_len), nlt,  					   cifs_sb->mnt_cifs_flags &  						CIFS_MOUNT_MAP_SPECIAL_CHR); +		pqst->len -= nls_nullsize(nlt);  	} else {  		pqst->name = filename;  		pqst->len = len; @@ -727,11 +728,12 @@ static int cifs_filldir(char *pfindEntry, struct file *file, filldir_t filldir,  		cifs_dir_info_to_fattr(&fattr, (FILE_DIRECTORY_INFO *)  					pfindEntry, cifs_sb); -	/* FIXME: make _to_fattr functions fill this out */ -	if (pCifsF->srch_inf.info_level == SMB_FIND_FILE_ID_FULL_DIR_INFO) +	if (inum && (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM)) {  		fattr.cf_uniqueid = inum; -	else +	} else {  		fattr.cf_uniqueid = iunique(sb, ROOT_I); +		cifs_autodisable_serverino(cifs_sb); +	}  	ino = cifs_uniqueid_to_ino_t(fattr.cf_uniqueid);  	tmp_dentry = cifs_readdir_lookup(file->f_dentry, &qstring, &fattr); | 
