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

📄 vfs_cifs.c

📁 samba最新软件
💻 C
📖 第 1 页 / 共 2 页
字号:
/*    Unix SMB/CIFS implementation.   CIFS-on-CIFS NTVFS filesystem backend   Copyright (C) Andrew Tridgell 2003   Copyright (C) James J Myers 2003 <myersjj@samba.org>   This program is free software; you can redistribute it and/or modify   it under the terms of the GNU General Public License as published by   the Free Software Foundation; either version 3 of the License, or   (at your option) any later version.      This program is distributed in the hope that it will be useful,   but WITHOUT ANY WARRANTY; without even the implied warranty of   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the   GNU General Public License for more details.      You should have received a copy of the GNU General Public License   along with this program.  If not, see <http://www.gnu.org/licenses/>.*//*  this implements a CIFS->CIFS NTVFS filesystem backend.   */#include "includes.h"#include "libcli/raw/libcliraw.h"#include "libcli/raw/raw_proto.h"#include "libcli/smb_composite/smb_composite.h"#include "auth/auth.h"#include "auth/credentials/credentials.h"#include "ntvfs/ntvfs.h"#include "lib/util/dlinklist.h"#include "param/param.h"#include "libcli/resolve/resolve.h"struct cvfs_file {	struct cvfs_file *prev, *next;	uint16_t fnum;	struct ntvfs_handle *h;};/* this is stored in ntvfs_private */struct cvfs_private {	struct smbcli_tree *tree;	struct smbcli_transport *transport;	struct ntvfs_module_context *ntvfs;	struct async_info *pending;	struct cvfs_file *files;	bool map_generic;	bool map_trans2;};/* a structure used to pass information to an async handler */struct async_info {	struct async_info *next, *prev;	struct cvfs_private *cvfs;	struct ntvfs_request *req;	struct smbcli_request *c_req;	struct cvfs_file *f;	void *parms;};#define SETUP_PID private->tree->session->pid = req->smbpid#define SETUP_FILE_HERE(f) do { \	f = ntvfs_handle_get_backend_data(io->generic.in.file.ntvfs, ntvfs); \	if (!f) return NT_STATUS_INVALID_HANDLE; \	io->generic.in.file.fnum = f->fnum; \} while (0)#define SETUP_FILE do { \	struct cvfs_file *f; \	SETUP_FILE_HERE(f); \} while (0)#define SETUP_PID_AND_FILE do { \	SETUP_PID; \	SETUP_FILE; \} while (0)#define CIFS_SERVER		"cifs:server"#define CIFS_USER		"cifs:user"#define CIFS_PASSWORD		"cifs:password"#define CIFS_DOMAIN		"cifs:domain"#define CIFS_SHARE		"cifs:share"#define CIFS_USE_MACHINE_ACCT	"cifs:use-machine-account"#define CIFS_MAP_GENERIC	"cifs:map-generic"#define CIFS_MAP_TRANS2		"cifs:map-trans2"#define CIFS_USE_MACHINE_ACCT_DEFAULT	false#define CIFS_MAP_GENERIC_DEFAULT	false#define CIFS_MAP_TRANS2_DEFAULT		true/*  a handler for oplock break events from the server - these need to be passed  along to the client */static bool oplock_handler(struct smbcli_transport *transport, uint16_t tid, uint16_t fnum, uint8_t level, void *p_private){	struct cvfs_private *private = p_private;	NTSTATUS status;	struct ntvfs_handle *h = NULL;	struct cvfs_file *f;	for (f=private->files; f; f=f->next) {		if (f->fnum != fnum) continue;		h = f->h;		break;	}	if (!h) {		DEBUG(5,("vfs_cifs: ignoring oplock break level %d for fnum %d\n", level, fnum));		return true;	}	DEBUG(5,("vfs_cifs: sending oplock break level %d for fnum %d\n", level, fnum));	status = ntvfs_send_oplock_break(private->ntvfs, h, level);	if (!NT_STATUS_IS_OK(status)) return false;	return true;}/*  connect to a share - used when a tree_connect operation comes in.*/static NTSTATUS cvfs_connect(struct ntvfs_module_context *ntvfs, 			     struct ntvfs_request *req, const char *sharename){	NTSTATUS status;	struct cvfs_private *private;	const char *host, *user, *pass, *domain, *remote_share;	struct smb_composite_connect io;	struct composite_context *creq;	struct share_config *scfg = ntvfs->ctx->config;	struct cli_credentials *credentials;	bool machine_account;	/* Here we need to determine which server to connect to.	 * For now we use parametric options, type cifs.	 * Later we will use security=server and auth_server.c.	 */	host = share_string_option(scfg, CIFS_SERVER, NULL);	user = share_string_option(scfg, CIFS_USER, NULL);	pass = share_string_option(scfg, CIFS_PASSWORD, NULL);	domain = share_string_option(scfg, CIFS_DOMAIN, NULL);	remote_share = share_string_option(scfg, CIFS_SHARE, NULL);	if (!remote_share) {		remote_share = sharename;	}	machine_account = share_bool_option(scfg, CIFS_USE_MACHINE_ACCT, CIFS_USE_MACHINE_ACCT_DEFAULT);	private = talloc_zero(ntvfs, struct cvfs_private);	if (!private) {		return NT_STATUS_NO_MEMORY;	}	ntvfs->private_data = private;	if (!host) {		DEBUG(1,("CIFS backend: You must supply server\n"));		return NT_STATUS_INVALID_PARAMETER;	} 		if (user && pass) {		DEBUG(5, ("CIFS backend: Using specified password\n"));		credentials = cli_credentials_init(private);		if (!credentials) {			return NT_STATUS_NO_MEMORY;		}		cli_credentials_set_conf(credentials, ntvfs->ctx->lp_ctx);		cli_credentials_set_username(credentials, user, CRED_SPECIFIED);		if (domain) {			cli_credentials_set_domain(credentials, domain, CRED_SPECIFIED);		}		cli_credentials_set_password(credentials, pass, CRED_SPECIFIED);	} else if (machine_account) {		DEBUG(5, ("CIFS backend: Using machine account\n"));		credentials = cli_credentials_init(private);		cli_credentials_set_conf(credentials, ntvfs->ctx->lp_ctx);		if (domain) {			cli_credentials_set_domain(credentials, domain, CRED_SPECIFIED);		}		status = cli_credentials_set_machine_account(credentials, ntvfs->ctx->lp_ctx);		if (!NT_STATUS_IS_OK(status)) {			return status;		}	} else if (req->session_info->credentials) {		DEBUG(5, ("CIFS backend: Using delegated credentials\n"));		credentials = req->session_info->credentials;	} else {		DEBUG(1,("CIFS backend: NO delegated credentials found: You must supply server, user and password or the client must supply delegated credentials\n"));		return NT_STATUS_INVALID_PARAMETER;	}	/* connect to the server, using the smbd event context */	io.in.dest_host = host;	io.in.dest_ports = lp_smb_ports(ntvfs->ctx->lp_ctx);	io.in.called_name = host;	io.in.credentials = credentials;	io.in.fallback_to_anonymous = false;	io.in.workgroup = lp_workgroup(ntvfs->ctx->lp_ctx);	io.in.service = remote_share;	io.in.service_type = "?????";	lp_smbcli_options(ntvfs->ctx->lp_ctx, &io.in.options);	if (!(ntvfs->ctx->client_caps & NTVFS_CLIENT_CAP_LEVEL_II_OPLOCKS)) {		io.in.options.use_level2_oplocks = false;	}	creq = smb_composite_connect_send(&io, private, 					  lp_resolve_context(ntvfs->ctx->lp_ctx),					  ntvfs->ctx->event_ctx);	status = smb_composite_connect_recv(creq, private);	NT_STATUS_NOT_OK_RETURN(status);	private->tree = io.out.tree;	private->transport = private->tree->session->transport;	SETUP_PID;	private->ntvfs = ntvfs;	ntvfs->ctx->fs_type = talloc_strdup(ntvfs->ctx, "NTFS");	NT_STATUS_HAVE_NO_MEMORY(ntvfs->ctx->fs_type);	ntvfs->ctx->dev_type = talloc_strdup(ntvfs->ctx, "A:");	NT_STATUS_HAVE_NO_MEMORY(ntvfs->ctx->dev_type);	/* we need to receive oplock break requests from the server */	smbcli_oplock_handler(private->transport, oplock_handler, private);	private->map_generic = share_bool_option(scfg, CIFS_MAP_GENERIC, CIFS_MAP_GENERIC_DEFAULT);	private->map_trans2 = share_bool_option(scfg, CIFS_MAP_TRANS2, CIFS_MAP_TRANS2_DEFAULT);	return NT_STATUS_OK;}/*  disconnect from a share*/static NTSTATUS cvfs_disconnect(struct ntvfs_module_context *ntvfs){	struct cvfs_private *private = ntvfs->private_data;	struct async_info *a, *an;	/* first cleanup pending requests */	for (a=private->pending; a; a = an) {		an = a->next;		smbcli_request_destroy(a->c_req);		talloc_free(a);	}	talloc_free(private);	ntvfs->private_data = NULL;	return NT_STATUS_OK;}/*  destroy an async info structure*/static int async_info_destructor(struct async_info *async){	DLIST_REMOVE(async->cvfs->pending, async);	return 0;}/*  a handler for simple async replies  this handler can only be used for functions that don't return any  parameters (those that just return a status code) */static void async_simple(struct smbcli_request *c_req){	struct async_info *async = c_req->async.private;	struct ntvfs_request *req = async->req;	req->async_states->status = smbcli_request_simple_recv(c_req);	talloc_free(async);	req->async_states->send_fn(req);}/* save some typing for the simple functions */#define ASYNC_RECV_TAIL_F(io, async_fn, file) do { \	if (!c_req) return NT_STATUS_UNSUCCESSFUL; \	{ \		struct async_info *async; \		async = talloc(req, struct async_info); \		if (!async) return NT_STATUS_NO_MEMORY; \		async->parms = io; \		async->req = req; \		async->f = file; \		async->cvfs = private; \		async->c_req = c_req; \		DLIST_ADD(private->pending, async); \		c_req->async.private = async; \		talloc_set_destructor(async, async_info_destructor); \	} \	c_req->async.fn = async_fn; \	req->async_states->state |= NTVFS_ASYNC_STATE_ASYNC; \	return NT_STATUS_OK; \} while (0)#define ASYNC_RECV_TAIL(io, async_fn) ASYNC_RECV_TAIL_F(io, async_fn, NULL)#define SIMPLE_ASYNC_TAIL ASYNC_RECV_TAIL(NULL, async_simple)/*  delete a file - the dirtype specifies the file types to include in the search.   The name can contain CIFS wildcards, but rarely does (except with OS/2 clients)*/static NTSTATUS cvfs_unlink(struct ntvfs_module_context *ntvfs, 			    struct ntvfs_request *req, union smb_unlink *unl){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	SETUP_PID;	/* see if the front end will allow us to perform this	   function asynchronously.  */	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		return smb_raw_unlink(private->tree, unl);	}	c_req = smb_raw_unlink_send(private->tree, unl);	SIMPLE_ASYNC_TAIL;}/*  a handler for async ioctl replies */static void async_ioctl(struct smbcli_request *c_req){	struct async_info *async = c_req->async.private;	struct ntvfs_request *req = async->req;	req->async_states->status = smb_raw_ioctl_recv(c_req, req, async->parms);	talloc_free(async);	req->async_states->send_fn(req);}/*  ioctl interface*/static NTSTATUS cvfs_ioctl(struct ntvfs_module_context *ntvfs, 			   struct ntvfs_request *req, union smb_ioctl *io){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	SETUP_PID_AND_FILE;	/* see if the front end will allow us to perform this	   function asynchronously.  */	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		return smb_raw_ioctl(private->tree, req, io);	}	c_req = smb_raw_ioctl_send(private->tree, io);	ASYNC_RECV_TAIL(io, async_ioctl);}/*  check if a directory exists*/static NTSTATUS cvfs_chkpath(struct ntvfs_module_context *ntvfs, 			     struct ntvfs_request *req, union smb_chkpath *cp){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	SETUP_PID;	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		return smb_raw_chkpath(private->tree, cp);	}	c_req = smb_raw_chkpath_send(private->tree, cp);	SIMPLE_ASYNC_TAIL;}/*  a handler for async qpathinfo replies */static void async_qpathinfo(struct smbcli_request *c_req){	struct async_info *async = c_req->async.private;	struct ntvfs_request *req = async->req;	req->async_states->status = smb_raw_pathinfo_recv(c_req, req, async->parms);	talloc_free(async);	req->async_states->send_fn(req);}/*  return info on a pathname*/static NTSTATUS cvfs_qpathinfo(struct ntvfs_module_context *ntvfs, 			       struct ntvfs_request *req, union smb_fileinfo *info){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	SETUP_PID;	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		return smb_raw_pathinfo(private->tree, req, info);	}	c_req = smb_raw_pathinfo_send(private->tree, info);	ASYNC_RECV_TAIL(info, async_qpathinfo);}/*  a handler for async qfileinfo replies */static void async_qfileinfo(struct smbcli_request *c_req){	struct async_info *async = c_req->async.private;	struct ntvfs_request *req = async->req;	req->async_states->status = smb_raw_fileinfo_recv(c_req, req, async->parms);	talloc_free(async);	req->async_states->send_fn(req);}/*  query info on a open file*/static NTSTATUS cvfs_qfileinfo(struct ntvfs_module_context *ntvfs, 			       struct ntvfs_request *req, union smb_fileinfo *io){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	SETUP_PID_AND_FILE;	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		return smb_raw_fileinfo(private->tree, req, io);	}	c_req = smb_raw_fileinfo_send(private->tree, io);	ASYNC_RECV_TAIL(io, async_qfileinfo);}/*  set info on a pathname*/static NTSTATUS cvfs_setpathinfo(struct ntvfs_module_context *ntvfs, 				 struct ntvfs_request *req, union smb_setfileinfo *st){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	SETUP_PID;	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		return smb_raw_setpathinfo(private->tree, st);	}	c_req = smb_raw_setpathinfo_send(private->tree, st);	SIMPLE_ASYNC_TAIL;}/*  a handler for async open replies */static void async_open(struct smbcli_request *c_req){	struct async_info *async = c_req->async.private;	struct cvfs_private *cvfs = async->cvfs;	struct ntvfs_request *req = async->req;	struct cvfs_file *f = async->f;	union smb_open *io = async->parms;	union smb_handle *file;	talloc_free(async);	req->async_states->status = smb_raw_open_recv(c_req, req, io);	SMB_OPEN_OUT_FILE(io, file);	f->fnum = file->fnum;	file->ntvfs = NULL;	if (!NT_STATUS_IS_OK(req->async_states->status)) goto failed;	req->async_states->status = ntvfs_handle_set_backend_data(f->h, cvfs->ntvfs, f);	if (!NT_STATUS_IS_OK(req->async_states->status)) goto failed;	file->ntvfs = f->h;	DLIST_ADD(cvfs->files, f);failed:	req->async_states->send_fn(req);}/*  open a file*/static NTSTATUS cvfs_open(struct ntvfs_module_context *ntvfs, 			  struct ntvfs_request *req, union smb_open *io){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	struct ntvfs_handle *h;	struct cvfs_file *f;	NTSTATUS status;	SETUP_PID;	if (io->generic.level != RAW_OPEN_GENERIC &&	    private->map_generic) {		return ntvfs_map_open(ntvfs, req, io);	}	status = ntvfs_handle_new(ntvfs, req, &h);	NT_STATUS_NOT_OK_RETURN(status);	f = talloc_zero(h, struct cvfs_file);	NT_STATUS_HAVE_NO_MEMORY(f);	f->h = h;	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		union smb_handle *file;		status = smb_raw_open(private->tree, req, io);		NT_STATUS_NOT_OK_RETURN(status);		SMB_OPEN_OUT_FILE(io, file);		f->fnum = file->fnum;		file->ntvfs = NULL;		status = ntvfs_handle_set_backend_data(f->h, private->ntvfs, f);		NT_STATUS_NOT_OK_RETURN(status);		file->ntvfs = f->h;		DLIST_ADD(private->files, f);		return NT_STATUS_OK;	}	c_req = smb_raw_open_send(private->tree, io);	ASYNC_RECV_TAIL_F(io, async_open, f);}/*  create a directory*/static NTSTATUS cvfs_mkdir(struct ntvfs_module_context *ntvfs, 			   struct ntvfs_request *req, union smb_mkdir *md){	struct cvfs_private *private = ntvfs->private_data;	struct smbcli_request *c_req;	SETUP_PID;	if (!(req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC)) {		return smb_raw_mkdir(private->tree, md);	}	c_req = smb_raw_mkdir_send(private->tree, md);	SIMPLE_ASYNC_TAIL;}/*  remove a directory*/static NTSTATUS cvfs_rmdir(struct ntvfs_module_context *ntvfs, 			   struct ntvfs_request *req, struct smb_rmdir *rd){	struct cvfs_private *private = ntvfs->private_data;

⌨️ 快捷键说明

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