mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	cifs: create SMB2_open_init()/SMB2_open_free() helpers.
Signed-off-by: Ronnie Sahlberg <lsahlber@redhat.com> Signed-off-by: Steve French <stfrench@microsoft.com> Reviewed-by: Paulo Alcantara <palcantara@suse.com> Reviewed-by: Pavel Shilovsky <pshilov@microsoft.com>
This commit is contained in:
		
							parent
							
								
									296ecbae7f
								
							
						
					
					
						commit
						1eb9fb5204
					
				
					 2 changed files with 78 additions and 71 deletions
				
			
		| 
						 | 
				
			
			@ -2052,43 +2052,27 @@ int smb311_posix_mkdir(const unsigned int xid, struct inode *inode,
 | 
			
		|||
}
 | 
			
		||||
 | 
			
		||||
int
 | 
			
		||||
SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		||||
	  __u8 *oplock, struct smb2_file_all_info *buf,
 | 
			
		||||
	  struct kvec *err_iov, int *buftype)
 | 
			
		||||
SMB2_open_init(struct cifs_tcon *tcon, struct smb_rqst *rqst, __u8 *oplock,
 | 
			
		||||
	       struct cifs_open_parms *oparms, __le16 *path)
 | 
			
		||||
{
 | 
			
		||||
	struct smb_rqst rqst;
 | 
			
		||||
	struct TCP_Server_Info *server = tcon->ses->server;
 | 
			
		||||
	struct smb2_create_req *req;
 | 
			
		||||
	struct smb2_create_rsp *rsp;
 | 
			
		||||
	struct TCP_Server_Info *server;
 | 
			
		||||
	struct cifs_tcon *tcon = oparms->tcon;
 | 
			
		||||
	struct cifs_ses *ses = tcon->ses;
 | 
			
		||||
	struct kvec iov[5]; /* make sure at least one for each open context */
 | 
			
		||||
	struct kvec rsp_iov = {NULL, 0};
 | 
			
		||||
	int resp_buftype;
 | 
			
		||||
	int uni_path_len;
 | 
			
		||||
	__le16 *copy_path = NULL;
 | 
			
		||||
	int copy_size;
 | 
			
		||||
	int rc = 0;
 | 
			
		||||
	unsigned int n_iov = 2;
 | 
			
		||||
	__u32 file_attributes = 0;
 | 
			
		||||
	char *dhc_buf = NULL, *lc_buf = NULL, *pc_buf = NULL;
 | 
			
		||||
	int flags = 0;
 | 
			
		||||
	int copy_size;
 | 
			
		||||
	int uni_path_len;
 | 
			
		||||
	unsigned int total_len;
 | 
			
		||||
 | 
			
		||||
	cifs_dbg(FYI, "create/open\n");
 | 
			
		||||
 | 
			
		||||
	if (ses && (ses->server))
 | 
			
		||||
		server = ses->server;
 | 
			
		||||
	else
 | 
			
		||||
		return -EIO;
 | 
			
		||||
	struct kvec *iov = rqst->rq_iov;
 | 
			
		||||
	__le16 *copy_path;
 | 
			
		||||
	int rc;
 | 
			
		||||
 | 
			
		||||
	rc = smb2_plain_req_init(SMB2_CREATE, tcon, (void **) &req, &total_len);
 | 
			
		||||
 | 
			
		||||
	if (rc)
 | 
			
		||||
		return rc;
 | 
			
		||||
 | 
			
		||||
	if (smb3_encryption_required(tcon))
 | 
			
		||||
		flags |= CIFS_TRANSFORM_REQ;
 | 
			
		||||
	iov[0].iov_base = (char *)req;
 | 
			
		||||
	/* -1 since last byte is buf[0] which is sent below (path) */
 | 
			
		||||
	iov[0].iov_len = total_len - 1;
 | 
			
		||||
 | 
			
		||||
	if (oparms->create_options & CREATE_OPTION_READONLY)
 | 
			
		||||
		file_attributes |= ATTR_READONLY;
 | 
			
		||||
| 
						 | 
				
			
			@ -2102,11 +2086,6 @@ SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		|||
	req->ShareAccess = FILE_SHARE_ALL_LE;
 | 
			
		||||
	req->CreateDisposition = cpu_to_le32(oparms->disposition);
 | 
			
		||||
	req->CreateOptions = cpu_to_le32(oparms->create_options & CREATE_OPTIONS_MASK);
 | 
			
		||||
 | 
			
		||||
	iov[0].iov_base = (char *)req;
 | 
			
		||||
	/* -1 since last byte is buf[0] which is sent below (path) */
 | 
			
		||||
	iov[0].iov_len = total_len - 1;
 | 
			
		||||
 | 
			
		||||
	req->NameOffset = cpu_to_le16(sizeof(struct smb2_create_req));
 | 
			
		||||
 | 
			
		||||
	/* [MS-SMB2] 2.2.13 NameOffset:
 | 
			
		||||
| 
						 | 
				
			
			@ -2124,10 +2103,8 @@ SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		|||
		rc = alloc_path_with_tree_prefix(©_path, ©_size,
 | 
			
		||||
						 &name_len,
 | 
			
		||||
						 tcon->treeName, path);
 | 
			
		||||
		if (rc) {
 | 
			
		||||
			cifs_small_buf_release(req);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
		}
 | 
			
		||||
		req->NameLength = cpu_to_le16(name_len * 2);
 | 
			
		||||
		uni_path_len = copy_size;
 | 
			
		||||
		path = copy_path;
 | 
			
		||||
| 
						 | 
				
			
			@ -2135,18 +2112,16 @@ SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		|||
		uni_path_len = (2 * UniStrnlen((wchar_t *)path, PATH_MAX)) + 2;
 | 
			
		||||
		/* MUST set path len (NameLength) to 0 opening root of share */
 | 
			
		||||
		req->NameLength = cpu_to_le16(uni_path_len - 2);
 | 
			
		||||
		if (uni_path_len % 8 != 0) {
 | 
			
		||||
			copy_size = roundup(uni_path_len, 8);
 | 
			
		||||
			copy_path = kzalloc(copy_size, GFP_KERNEL);
 | 
			
		||||
			if (!copy_path) {
 | 
			
		||||
				cifs_small_buf_release(req);
 | 
			
		||||
				return -ENOMEM;
 | 
			
		||||
			}
 | 
			
		||||
			memcpy((char *)copy_path, (const char *)path,
 | 
			
		||||
			       uni_path_len);
 | 
			
		||||
			uni_path_len = copy_size;
 | 
			
		||||
			path = copy_path;
 | 
			
		||||
		}
 | 
			
		||||
		copy_size = uni_path_len;
 | 
			
		||||
		if (copy_size % 8 != 0)
 | 
			
		||||
			copy_size = roundup(copy_size, 8);
 | 
			
		||||
		copy_path = kzalloc(copy_size, GFP_KERNEL);
 | 
			
		||||
		if (!copy_path)
 | 
			
		||||
			return -ENOMEM;
 | 
			
		||||
		memcpy((char *)copy_path, (const char *)path,
 | 
			
		||||
		       uni_path_len);
 | 
			
		||||
		uni_path_len = copy_size;
 | 
			
		||||
		path = copy_path;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	iov[1].iov_len = uni_path_len;
 | 
			
		||||
| 
						 | 
				
			
			@ -2161,12 +2136,8 @@ SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		|||
	else {
 | 
			
		||||
		rc = add_lease_context(server, iov, &n_iov,
 | 
			
		||||
				       oparms->fid->lease_key, oplock);
 | 
			
		||||
		if (rc) {
 | 
			
		||||
			cifs_small_buf_release(req);
 | 
			
		||||
			kfree(copy_path);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
		}
 | 
			
		||||
		lc_buf = iov[n_iov-1].iov_base;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (*oplock == SMB2_OPLOCK_LEVEL_BATCH) {
 | 
			
		||||
| 
						 | 
				
			
			@ -2180,13 +2151,8 @@ SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		|||
 | 
			
		||||
		rc = add_durable_context(iov, &n_iov, oparms,
 | 
			
		||||
					tcon->use_persistent);
 | 
			
		||||
		if (rc) {
 | 
			
		||||
			cifs_small_buf_release(req);
 | 
			
		||||
			kfree(copy_path);
 | 
			
		||||
			kfree(lc_buf);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
		}
 | 
			
		||||
		dhc_buf = iov[n_iov-1].iov_base;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (tcon->posix_extensions) {
 | 
			
		||||
| 
						 | 
				
			
			@ -2198,23 +2164,63 @@ SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		rc = add_posix_context(iov, &n_iov, oparms->mode);
 | 
			
		||||
		if (rc) {
 | 
			
		||||
			cifs_small_buf_release(req);
 | 
			
		||||
			kfree(copy_path);
 | 
			
		||||
			kfree(lc_buf);
 | 
			
		||||
			kfree(dhc_buf);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
		}
 | 
			
		||||
		pc_buf = iov[n_iov-1].iov_base;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	rqst->rq_nvec = n_iov;
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* rq_iov[0] is the request and is released by cifs_small_buf_release().
 | 
			
		||||
 * All other vectors are freed by kfree().
 | 
			
		||||
 */
 | 
			
		||||
void
 | 
			
		||||
SMB2_open_free(struct smb_rqst *rqst)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	cifs_small_buf_release(rqst->rq_iov[0].iov_base);
 | 
			
		||||
	for (i = 1; i < rqst->rq_nvec; i++)
 | 
			
		||||
		kfree(rqst->rq_iov[i].iov_base);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int
 | 
			
		||||
SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		||||
	  __u8 *oplock, struct smb2_file_all_info *buf,
 | 
			
		||||
	  struct kvec *err_iov, int *buftype)
 | 
			
		||||
{
 | 
			
		||||
	struct smb_rqst rqst;
 | 
			
		||||
	struct smb2_create_rsp *rsp = NULL;
 | 
			
		||||
	struct TCP_Server_Info *server;
 | 
			
		||||
	struct cifs_tcon *tcon = oparms->tcon;
 | 
			
		||||
	struct cifs_ses *ses = tcon->ses;
 | 
			
		||||
	struct kvec iov[5]; /* make sure at least one for each open context */
 | 
			
		||||
	struct kvec rsp_iov = {NULL, 0};
 | 
			
		||||
	int resp_buftype;
 | 
			
		||||
	int rc = 0;
 | 
			
		||||
	int flags = 0;
 | 
			
		||||
 | 
			
		||||
	cifs_dbg(FYI, "create/open\n");
 | 
			
		||||
	if (ses && (ses->server))
 | 
			
		||||
		server = ses->server;
 | 
			
		||||
	else
 | 
			
		||||
		return -EIO;
 | 
			
		||||
 | 
			
		||||
	if (smb3_encryption_required(tcon))
 | 
			
		||||
		flags |= CIFS_TRANSFORM_REQ;
 | 
			
		||||
 | 
			
		||||
	memset(&rqst, 0, sizeof(struct smb_rqst));
 | 
			
		||||
	memset(&iov, 0, sizeof(iov));
 | 
			
		||||
	rqst.rq_iov = iov;
 | 
			
		||||
	rqst.rq_nvec = n_iov;
 | 
			
		||||
	rqst.rq_nvec = 5;
 | 
			
		||||
 | 
			
		||||
	rc = SMB2_open_init(tcon, &rqst, oplock, oparms, path);
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto creat_exit;
 | 
			
		||||
 | 
			
		||||
	rc = cifs_send_recv(xid, ses, &rqst, &resp_buftype, flags,
 | 
			
		||||
			    &rsp_iov);
 | 
			
		||||
	cifs_small_buf_release(req);
 | 
			
		||||
	rsp = (struct smb2_create_rsp *)rsp_iov.iov_base;
 | 
			
		||||
 | 
			
		||||
	if (rc != 0) {
 | 
			
		||||
| 
						 | 
				
			
			@ -2251,10 +2257,7 @@ SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms, __le16 *path,
 | 
			
		|||
	else
 | 
			
		||||
		*oplock = rsp->OplockLevel;
 | 
			
		||||
creat_exit:
 | 
			
		||||
	kfree(copy_path);
 | 
			
		||||
	kfree(lc_buf);
 | 
			
		||||
	kfree(dhc_buf);
 | 
			
		||||
	kfree(pc_buf);
 | 
			
		||||
	SMB2_open_free(&rqst);
 | 
			
		||||
	free_rsp_buf(resp_buftype, rsp);
 | 
			
		||||
	return rc;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -133,6 +133,10 @@ extern int SMB2_open(const unsigned int xid, struct cifs_open_parms *oparms,
 | 
			
		|||
		     __le16 *path, __u8 *oplock,
 | 
			
		||||
		     struct smb2_file_all_info *buf,
 | 
			
		||||
		     struct kvec *err_iov, int *resp_buftype);
 | 
			
		||||
extern int SMB2_open_init(struct cifs_tcon *tcon, struct smb_rqst *rqst,
 | 
			
		||||
			  __u8 *oplock, struct cifs_open_parms *oparms,
 | 
			
		||||
			  __le16 *path);
 | 
			
		||||
extern void SMB2_open_free(struct smb_rqst *rqst);
 | 
			
		||||
extern int SMB2_ioctl(const unsigned int xid, struct cifs_tcon *tcon,
 | 
			
		||||
		     u64 persistent_fid, u64 volatile_fid, u32 opcode,
 | 
			
		||||
		     bool is_fsctl, char *in_data, u32 indatalen,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue