mirror of
https://github.com/samba-team/samba.git
synced 2025-01-11 05:18:09 +03:00
8899eb21d4
The original implementation of the virtual xattr get_real_filename in gluster was misusing the ENOENT errno as the authoritative anwer that the file/dir that we were asking the real filename for does not exist. But since the getxattr call is done on the parent directory, this is a violation of the getxattr API which uses ENOENT for the case that the file/dir that the getxattr call is done against does not exist. Now after a recent regression for fuse-mount re-exports due to gluster mapping ENOENT to ESTALE in the fuse-bridge, the gluster implementation is changed to more correctly return ENOATTR if the requested file does not exist. This patch changes the glusterfs vfs module to treat ENOATTR as ENOENT to be fully functional again with latest gluster. - Without this patch, samba against a new gluster will work correctly, but the get_real_filename optimization for a non-existing entry is lost. - With this patch, Samba will not work correctly any more against very old gluster servers: Those (correctly) returned ENOATTR always, which Samba originally interpreted as EOPNOTSUPP, triggering the expensive directory scan. With this patch, ENOATTR is interpreted as ENOENT, the authoritative answer that the requested entry does not exist, which is wrong unless it really does not exist. Signed-off-by: Michael Adam <obnox@samba.org> Reviewed-by: Guenther Deschner <gd@samba.org>
1712 lines
40 KiB
C
1712 lines
40 KiB
C
/*
|
|
Unix SMB/CIFS implementation.
|
|
|
|
Wrap GlusterFS GFAPI calls in vfs functions.
|
|
|
|
Copyright (c) 2013 Anand Avati <avati@redhat.com>
|
|
|
|
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/>.
|
|
*/
|
|
|
|
/**
|
|
* @file vfs_glusterfs.c
|
|
* @author Anand Avati <avati@redhat.com>
|
|
* @date May 2013
|
|
* @brief Samba VFS module for glusterfs
|
|
*
|
|
* @todo
|
|
* - sendfile/recvfile support
|
|
*
|
|
* A Samba VFS module for GlusterFS, based on Gluster's libgfapi.
|
|
* This is a "bottom" vfs module (not something to be stacked on top of
|
|
* another module), and translates (most) calls to the closest actions
|
|
* available in libgfapi.
|
|
*
|
|
*/
|
|
|
|
#include "includes.h"
|
|
#include "smbd/smbd.h"
|
|
#include <stdio.h>
|
|
#include <glusterfs/api/glfs.h>
|
|
#include "lib/util/dlinklist.h"
|
|
#include "lib/util/tevent_unix.h"
|
|
#include "smbd/globals.h"
|
|
#include "lib/util/sys_rw.h"
|
|
#include "smbprofile.h"
|
|
#include "modules/posixacl_xattr.h"
|
|
|
|
#define DEFAULT_VOLFILE_SERVER "localhost"
|
|
#define GLUSTER_NAME_MAX 255
|
|
|
|
static int read_fd = -1;
|
|
static int write_fd = -1;
|
|
static struct tevent_fd *aio_read_event = NULL;
|
|
|
|
/**
|
|
* Helper to convert struct stat to struct stat_ex.
|
|
*/
|
|
static void smb_stat_ex_from_stat(struct stat_ex *dst, const struct stat *src)
|
|
{
|
|
ZERO_STRUCTP(dst);
|
|
|
|
dst->st_ex_dev = src->st_dev;
|
|
dst->st_ex_ino = src->st_ino;
|
|
dst->st_ex_mode = src->st_mode;
|
|
dst->st_ex_nlink = src->st_nlink;
|
|
dst->st_ex_uid = src->st_uid;
|
|
dst->st_ex_gid = src->st_gid;
|
|
dst->st_ex_rdev = src->st_rdev;
|
|
dst->st_ex_size = src->st_size;
|
|
dst->st_ex_atime.tv_sec = src->st_atime;
|
|
dst->st_ex_mtime.tv_sec = src->st_mtime;
|
|
dst->st_ex_ctime.tv_sec = src->st_ctime;
|
|
dst->st_ex_btime.tv_sec = src->st_mtime;
|
|
dst->st_ex_blksize = src->st_blksize;
|
|
dst->st_ex_blocks = src->st_blocks;
|
|
#ifdef STAT_HAVE_NSEC
|
|
dst->st_ex_atime.tv_nsec = src->st_atime_nsec;
|
|
dst->st_ex_mtime.tv_nsec = src->st_mtime_nsec;
|
|
dst->st_ex_ctime.tv_nsec = src->st_ctime_nsec;
|
|
dst->st_ex_btime.tv_nsec = src->st_mtime_nsec;
|
|
#endif
|
|
}
|
|
|
|
/* pre-opened glfs_t */
|
|
|
|
static struct glfs_preopened {
|
|
char *volume;
|
|
char *connectpath;
|
|
glfs_t *fs;
|
|
int ref;
|
|
struct glfs_preopened *next, *prev;
|
|
} *glfs_preopened;
|
|
|
|
|
|
static int glfs_set_preopened(const char *volume, const char *connectpath, glfs_t *fs)
|
|
{
|
|
struct glfs_preopened *entry = NULL;
|
|
|
|
entry = talloc_zero(NULL, struct glfs_preopened);
|
|
if (!entry) {
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
|
|
entry->volume = talloc_strdup(entry, volume);
|
|
if (!entry->volume) {
|
|
talloc_free(entry);
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
|
|
entry->connectpath = talloc_strdup(entry, connectpath);
|
|
if (entry->connectpath == NULL) {
|
|
talloc_free(entry);
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
|
|
entry->fs = fs;
|
|
entry->ref = 1;
|
|
|
|
DLIST_ADD(glfs_preopened, entry);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static glfs_t *glfs_find_preopened(const char *volume, const char *connectpath)
|
|
{
|
|
struct glfs_preopened *entry = NULL;
|
|
|
|
for (entry = glfs_preopened; entry; entry = entry->next) {
|
|
if (strcmp(entry->volume, volume) == 0 &&
|
|
strcmp(entry->connectpath, connectpath) == 0)
|
|
{
|
|
entry->ref++;
|
|
return entry->fs;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void glfs_clear_preopened(glfs_t *fs)
|
|
{
|
|
struct glfs_preopened *entry = NULL;
|
|
|
|
for (entry = glfs_preopened; entry; entry = entry->next) {
|
|
if (entry->fs == fs) {
|
|
if (--entry->ref)
|
|
return;
|
|
|
|
DLIST_REMOVE(glfs_preopened, entry);
|
|
|
|
glfs_fini(entry->fs);
|
|
talloc_free(entry);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int vfs_gluster_set_volfile_servers(glfs_t *fs,
|
|
const char *volfile_servers)
|
|
{
|
|
char *server = NULL;
|
|
size_t server_count = 0;
|
|
size_t server_success = 0;
|
|
int ret = -1;
|
|
TALLOC_CTX *frame = talloc_stackframe();
|
|
|
|
DBG_INFO("servers list %s\n", volfile_servers);
|
|
|
|
while (next_token_talloc(frame, &volfile_servers, &server, " \t")) {
|
|
char *transport = NULL;
|
|
char *host = NULL;
|
|
int port = 0;
|
|
|
|
server_count++;
|
|
DBG_INFO("server %zu %s\n", server_count, server);
|
|
|
|
/* Determine the transport type */
|
|
if (strncmp(server, "unix+", 5) == 0) {
|
|
port = 0;
|
|
transport = talloc_strdup(frame, "unix");
|
|
if (!transport) {
|
|
errno = ENOMEM;
|
|
goto out;
|
|
}
|
|
host = talloc_strdup(frame, server + 5);
|
|
if (!host) {
|
|
errno = ENOMEM;
|
|
goto out;
|
|
}
|
|
} else {
|
|
char *p = NULL;
|
|
char *port_index = NULL;
|
|
|
|
if (strncmp(server, "tcp+", 4) == 0) {
|
|
server += 4;
|
|
}
|
|
|
|
/* IPv6 is enclosed in []
|
|
* ':' before ']' is part of IPv6
|
|
* ':' after ']' indicates port
|
|
*/
|
|
p = server;
|
|
if (server[0] == '[') {
|
|
server++;
|
|
p = index(server, ']');
|
|
if (p == NULL) {
|
|
/* Malformed IPv6 */
|
|
continue;
|
|
}
|
|
p[0] = '\0';
|
|
p++;
|
|
}
|
|
|
|
port_index = index(p, ':');
|
|
|
|
if (port_index == NULL) {
|
|
port = 0;
|
|
} else {
|
|
port = atoi(port_index + 1);
|
|
port_index[0] = '\0';
|
|
}
|
|
transport = talloc_strdup(frame, "tcp");
|
|
if (!transport) {
|
|
errno = ENOMEM;
|
|
goto out;
|
|
}
|
|
host = talloc_strdup(frame, server);
|
|
if (!host) {
|
|
errno = ENOMEM;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
DBG_INFO("Calling set volfile server with params "
|
|
"transport=%s, host=%s, port=%d\n", transport,
|
|
host, port);
|
|
|
|
ret = glfs_set_volfile_server(fs, transport, host, port);
|
|
if (ret < 0) {
|
|
DBG_WARNING("Failed to set volfile_server "
|
|
"transport=%s, host=%s, port=%d (%s)\n",
|
|
transport, host, port, strerror(errno));
|
|
} else {
|
|
server_success++;
|
|
}
|
|
}
|
|
|
|
out:
|
|
if (server_count == 0) {
|
|
ret = -1;
|
|
} else if (server_success < server_count) {
|
|
DBG_WARNING("Failed to set %zu out of %zu servers parsed\n",
|
|
server_count - server_success, server_count);
|
|
ret = 0;
|
|
}
|
|
|
|
TALLOC_FREE(frame);
|
|
return ret;
|
|
}
|
|
|
|
/* Disk Operations */
|
|
|
|
static int vfs_gluster_connect(struct vfs_handle_struct *handle,
|
|
const char *service,
|
|
const char *user)
|
|
{
|
|
const char *volfile_servers;
|
|
const char *volume;
|
|
char *logfile;
|
|
int loglevel;
|
|
glfs_t *fs = NULL;
|
|
TALLOC_CTX *tmp_ctx;
|
|
int ret = 0;
|
|
|
|
tmp_ctx = talloc_new(NULL);
|
|
if (tmp_ctx == NULL) {
|
|
ret = -1;
|
|
goto done;
|
|
}
|
|
logfile = lp_parm_talloc_string(tmp_ctx, SNUM(handle->conn), "glusterfs",
|
|
"logfile", NULL);
|
|
|
|
loglevel = lp_parm_int(SNUM(handle->conn), "glusterfs", "loglevel", -1);
|
|
|
|
volfile_servers = lp_parm_talloc_string(tmp_ctx, SNUM(handle->conn),
|
|
"glusterfs", "volfile_server",
|
|
NULL);
|
|
if (volfile_servers == NULL) {
|
|
volfile_servers = DEFAULT_VOLFILE_SERVER;
|
|
}
|
|
|
|
volume = lp_parm_const_string(SNUM(handle->conn), "glusterfs", "volume",
|
|
NULL);
|
|
if (volume == NULL) {
|
|
volume = service;
|
|
}
|
|
|
|
fs = glfs_find_preopened(volume, handle->conn->connectpath);
|
|
if (fs) {
|
|
goto done;
|
|
}
|
|
|
|
fs = glfs_new(volume);
|
|
if (fs == NULL) {
|
|
ret = -1;
|
|
goto done;
|
|
}
|
|
|
|
ret = vfs_gluster_set_volfile_servers(fs, volfile_servers);
|
|
if (ret < 0) {
|
|
DBG_ERR("Failed to set volfile_servers from list %s\n",
|
|
volfile_servers);
|
|
goto done;
|
|
}
|
|
|
|
ret = glfs_set_xlator_option(fs, "*-md-cache", "cache-posix-acl",
|
|
"true");
|
|
if (ret < 0) {
|
|
DEBUG(0, ("%s: Failed to set xlator options\n", volume));
|
|
goto done;
|
|
}
|
|
|
|
|
|
ret = glfs_set_xlator_option(fs, "*-snapview-client",
|
|
"snapdir-entry-path",
|
|
handle->conn->connectpath);
|
|
if (ret < 0) {
|
|
DEBUG(0, ("%s: Failed to set xlator option:"
|
|
" snapdir-entry-path\n", volume));
|
|
goto done;
|
|
}
|
|
|
|
ret = glfs_set_logging(fs, logfile, loglevel);
|
|
if (ret < 0) {
|
|
DEBUG(0, ("%s: Failed to set logfile %s loglevel %d\n",
|
|
volume, logfile, loglevel));
|
|
goto done;
|
|
}
|
|
|
|
ret = glfs_init(fs);
|
|
if (ret < 0) {
|
|
DEBUG(0, ("%s: Failed to initialize volume (%s)\n",
|
|
volume, strerror(errno)));
|
|
goto done;
|
|
}
|
|
|
|
ret = glfs_set_preopened(volume, handle->conn->connectpath, fs);
|
|
if (ret < 0) {
|
|
DEBUG(0, ("%s: Failed to register volume (%s)\n",
|
|
volume, strerror(errno)));
|
|
goto done;
|
|
}
|
|
|
|
/*
|
|
* The shadow_copy2 module will fail to export subdirectories
|
|
* of a gluster volume unless we specify the mount point,
|
|
* because the detection fails if the file system is not
|
|
* locally mounted:
|
|
* https://bugzilla.samba.org/show_bug.cgi?id=13091
|
|
*/
|
|
lp_do_parameter(SNUM(handle->conn), "shadow:mountpoint", "/");
|
|
|
|
/*
|
|
* Unless we have an async implementation of getxattrat turn this off.
|
|
*/
|
|
lp_do_parameter(SNUM(handle->conn), "smbd:async dosmode", "false");
|
|
|
|
done:
|
|
if (ret < 0) {
|
|
if (fs)
|
|
glfs_fini(fs);
|
|
} else {
|
|
DBG_ERR("%s: Initialized volume from servers %s\n",
|
|
volume, volfile_servers);
|
|
handle->data = fs;
|
|
}
|
|
talloc_free(tmp_ctx);
|
|
return ret;
|
|
}
|
|
|
|
static void vfs_gluster_disconnect(struct vfs_handle_struct *handle)
|
|
{
|
|
glfs_t *fs = NULL;
|
|
|
|
fs = handle->data;
|
|
|
|
glfs_clear_preopened(fs);
|
|
}
|
|
|
|
static uint64_t vfs_gluster_disk_free(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
uint64_t *bsize_p,
|
|
uint64_t *dfree_p,
|
|
uint64_t *dsize_p)
|
|
{
|
|
struct statvfs statvfs = { 0, };
|
|
int ret;
|
|
|
|
ret = glfs_statvfs(handle->data, smb_fname->base_name, &statvfs);
|
|
if (ret < 0) {
|
|
return -1;
|
|
}
|
|
|
|
if (bsize_p != NULL) {
|
|
*bsize_p = (uint64_t)statvfs.f_bsize; /* Block size */
|
|
}
|
|
if (dfree_p != NULL) {
|
|
*dfree_p = (uint64_t)statvfs.f_bavail; /* Available Block units */
|
|
}
|
|
if (dsize_p != NULL) {
|
|
*dsize_p = (uint64_t)statvfs.f_blocks; /* Total Block units */
|
|
}
|
|
|
|
return (uint64_t)statvfs.f_bavail;
|
|
}
|
|
|
|
static int vfs_gluster_get_quota(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
enum SMB_QUOTA_TYPE qtype,
|
|
unid_t id,
|
|
SMB_DISK_QUOTA *qt)
|
|
{
|
|
errno = ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
static int
|
|
vfs_gluster_set_quota(struct vfs_handle_struct *handle,
|
|
enum SMB_QUOTA_TYPE qtype, unid_t id, SMB_DISK_QUOTA *qt)
|
|
{
|
|
errno = ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
static int vfs_gluster_statvfs(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
struct vfs_statvfs_struct *vfs_statvfs)
|
|
{
|
|
struct statvfs statvfs = { 0, };
|
|
int ret;
|
|
|
|
ret = glfs_statvfs(handle->data, smb_fname->base_name, &statvfs);
|
|
if (ret < 0) {
|
|
DEBUG(0, ("glfs_statvfs(%s) failed: %s\n",
|
|
smb_fname->base_name, strerror(errno)));
|
|
return -1;
|
|
}
|
|
|
|
ZERO_STRUCTP(vfs_statvfs);
|
|
|
|
vfs_statvfs->OptimalTransferSize = statvfs.f_frsize;
|
|
vfs_statvfs->BlockSize = statvfs.f_bsize;
|
|
vfs_statvfs->TotalBlocks = statvfs.f_blocks;
|
|
vfs_statvfs->BlocksAvail = statvfs.f_bfree;
|
|
vfs_statvfs->UserBlocksAvail = statvfs.f_bavail;
|
|
vfs_statvfs->TotalFileNodes = statvfs.f_files;
|
|
vfs_statvfs->FreeFileNodes = statvfs.f_ffree;
|
|
vfs_statvfs->FsIdentifier = statvfs.f_fsid;
|
|
vfs_statvfs->FsCapabilities =
|
|
FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static uint32_t vfs_gluster_fs_capabilities(struct vfs_handle_struct *handle,
|
|
enum timestamp_set_resolution *p_ts_res)
|
|
{
|
|
uint32_t caps = FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES;
|
|
|
|
#ifdef HAVE_GFAPI_VER_6
|
|
caps |= FILE_SUPPORTS_SPARSE_FILES;
|
|
#endif
|
|
|
|
#ifdef STAT_HAVE_NSEC
|
|
*p_ts_res = TIMESTAMP_SET_NT_OR_BETTER;
|
|
#endif
|
|
|
|
return caps;
|
|
}
|
|
|
|
static DIR *vfs_gluster_opendir(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
const char *mask,
|
|
uint32_t attributes)
|
|
{
|
|
glfs_fd_t *fd;
|
|
|
|
fd = glfs_opendir(handle->data, smb_fname->base_name);
|
|
if (fd == NULL) {
|
|
DEBUG(0, ("glfs_opendir(%s) failed: %s\n",
|
|
smb_fname->base_name, strerror(errno)));
|
|
}
|
|
|
|
return (DIR *) fd;
|
|
}
|
|
|
|
static glfs_fd_t *vfs_gluster_fetch_glfd(struct vfs_handle_struct *handle,
|
|
files_struct *fsp)
|
|
{
|
|
glfs_fd_t **glfd = (glfs_fd_t **)VFS_FETCH_FSP_EXTENSION(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_INFO("Failed to fetch fsp extension\n");
|
|
return NULL;
|
|
}
|
|
if (*glfd == NULL) {
|
|
DBG_INFO("Empty glfs_fd_t pointer\n");
|
|
return NULL;
|
|
}
|
|
|
|
return *glfd;
|
|
}
|
|
|
|
static DIR *vfs_gluster_fdopendir(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, const char *mask,
|
|
uint32_t attributes)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return NULL;
|
|
}
|
|
|
|
return (DIR *)glfd;
|
|
}
|
|
|
|
static int vfs_gluster_closedir(struct vfs_handle_struct *handle, DIR *dirp)
|
|
{
|
|
return glfs_closedir((void *)dirp);
|
|
}
|
|
|
|
static struct dirent *vfs_gluster_readdir(struct vfs_handle_struct *handle,
|
|
DIR *dirp, SMB_STRUCT_STAT *sbuf)
|
|
{
|
|
static char direntbuf[512];
|
|
int ret;
|
|
struct stat stat;
|
|
struct dirent *dirent = 0;
|
|
|
|
if (sbuf != NULL) {
|
|
ret = glfs_readdirplus_r((void *)dirp, &stat, (void *)direntbuf,
|
|
&dirent);
|
|
} else {
|
|
ret = glfs_readdir_r((void *)dirp, (void *)direntbuf, &dirent);
|
|
}
|
|
|
|
if ((ret < 0) || (dirent == NULL)) {
|
|
return NULL;
|
|
}
|
|
|
|
if (sbuf != NULL) {
|
|
smb_stat_ex_from_stat(sbuf, &stat);
|
|
}
|
|
|
|
return dirent;
|
|
}
|
|
|
|
static long vfs_gluster_telldir(struct vfs_handle_struct *handle, DIR *dirp)
|
|
{
|
|
return glfs_telldir((void *)dirp);
|
|
}
|
|
|
|
static void vfs_gluster_seekdir(struct vfs_handle_struct *handle, DIR *dirp,
|
|
long offset)
|
|
{
|
|
glfs_seekdir((void *)dirp, offset);
|
|
}
|
|
|
|
static void vfs_gluster_rewinddir(struct vfs_handle_struct *handle, DIR *dirp)
|
|
{
|
|
glfs_seekdir((void *)dirp, 0);
|
|
}
|
|
|
|
static int vfs_gluster_mkdir(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
mode_t mode)
|
|
{
|
|
return glfs_mkdir(handle->data, smb_fname->base_name, mode);
|
|
}
|
|
|
|
static int vfs_gluster_rmdir(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname)
|
|
{
|
|
return glfs_rmdir(handle->data, smb_fname->base_name);
|
|
}
|
|
|
|
static int vfs_gluster_open(struct vfs_handle_struct *handle,
|
|
struct smb_filename *smb_fname, files_struct *fsp,
|
|
int flags, mode_t mode)
|
|
{
|
|
glfs_fd_t *glfd;
|
|
glfs_fd_t **p_tmp;
|
|
|
|
p_tmp = VFS_ADD_FSP_EXTENSION(handle, fsp, glfs_fd_t *, NULL);
|
|
if (p_tmp == NULL) {
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
|
|
if (flags & O_DIRECTORY) {
|
|
glfd = glfs_opendir(handle->data, smb_fname->base_name);
|
|
} else if (flags & O_CREAT) {
|
|
glfd = glfs_creat(handle->data, smb_fname->base_name, flags,
|
|
mode);
|
|
} else {
|
|
glfd = glfs_open(handle->data, smb_fname->base_name, flags);
|
|
}
|
|
|
|
if (glfd == NULL) {
|
|
/* no extension destroy_fn, so no need to save errno */
|
|
VFS_REMOVE_FSP_EXTENSION(handle, fsp);
|
|
return -1;
|
|
}
|
|
|
|
*p_tmp = glfd;
|
|
/* An arbitrary value for error reporting, so you know its us. */
|
|
return 13371337;
|
|
}
|
|
|
|
static int vfs_gluster_close(struct vfs_handle_struct *handle,
|
|
files_struct *fsp)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
VFS_REMOVE_FSP_EXTENSION(handle, fsp);
|
|
return glfs_close(glfd);
|
|
}
|
|
|
|
static ssize_t vfs_gluster_pread(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, void *data, size_t n,
|
|
off_t offset)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
#ifdef HAVE_GFAPI_VER_7_6
|
|
return glfs_pread(glfd, data, n, offset, 0, NULL);
|
|
#else
|
|
return glfs_pread(glfd, data, n, offset, 0);
|
|
#endif
|
|
}
|
|
|
|
struct glusterfs_aio_state;
|
|
|
|
struct glusterfs_aio_wrapper {
|
|
struct glusterfs_aio_state *state;
|
|
};
|
|
|
|
struct glusterfs_aio_state {
|
|
ssize_t ret;
|
|
struct tevent_req *req;
|
|
bool cancelled;
|
|
struct vfs_aio_state vfs_aio_state;
|
|
struct timespec start;
|
|
};
|
|
|
|
static int aio_wrapper_destructor(struct glusterfs_aio_wrapper *wrap)
|
|
{
|
|
if (wrap->state != NULL) {
|
|
wrap->state->cancelled = true;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* This function is the callback that will be called on glusterfs
|
|
* threads once the async IO submitted is complete. To notify
|
|
* Samba of the completion we use a pipe based queue.
|
|
*/
|
|
#ifdef HAVE_GFAPI_VER_7_6
|
|
static void aio_glusterfs_done(glfs_fd_t *fd, ssize_t ret,
|
|
struct glfs_stat *prestat,
|
|
struct glfs_stat *poststat,
|
|
void *data)
|
|
#else
|
|
static void aio_glusterfs_done(glfs_fd_t *fd, ssize_t ret, void *data)
|
|
#endif
|
|
{
|
|
struct glusterfs_aio_state *state = NULL;
|
|
int sts = 0;
|
|
struct timespec end;
|
|
|
|
state = (struct glusterfs_aio_state *)data;
|
|
|
|
PROFILE_TIMESTAMP(&end);
|
|
|
|
if (ret < 0) {
|
|
state->ret = -1;
|
|
state->vfs_aio_state.error = errno;
|
|
} else {
|
|
state->ret = ret;
|
|
}
|
|
state->vfs_aio_state.duration = nsec_time_diff(&end, &state->start);
|
|
|
|
/*
|
|
* Write the state pointer to glusterfs_aio_state to the
|
|
* pipe, so we can call tevent_req_done() from the main thread,
|
|
* because tevent_req_done() is not designed to be executed in
|
|
* the multithread environment, so tevent_req_done() must be
|
|
* executed from the smbd main thread.
|
|
*
|
|
* write(2) on pipes with sizes under _POSIX_PIPE_BUF
|
|
* in size is atomic, without this, the use op pipes in this
|
|
* code would not work.
|
|
*
|
|
* sys_write is a thin enough wrapper around write(2)
|
|
* that we can trust it here.
|
|
*/
|
|
|
|
sts = sys_write(write_fd, &state, sizeof(struct glusterfs_aio_state *));
|
|
if (sts < 0) {
|
|
DEBUG(0,("\nWrite to pipe failed (%s)", strerror(errno)));
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Read each req off the pipe and process it.
|
|
*/
|
|
static void aio_tevent_fd_done(struct tevent_context *event_ctx,
|
|
struct tevent_fd *fde,
|
|
uint16_t flags, void *data)
|
|
{
|
|
struct tevent_req *req = NULL;
|
|
struct glusterfs_aio_state *state = NULL;
|
|
int sts = 0;
|
|
|
|
/*
|
|
* read(2) on pipes is atomic if the needed data is available
|
|
* in the pipe, per SUS and POSIX. Because we always write
|
|
* to the pipe in sizeof(struct tevent_req *) chunks, we can
|
|
* always read in those chunks, atomically.
|
|
*
|
|
* sys_read is a thin enough wrapper around read(2) that we
|
|
* can trust it here.
|
|
*/
|
|
|
|
sts = sys_read(read_fd, &state, sizeof(struct glusterfs_aio_state *));
|
|
|
|
if (sts < 0) {
|
|
DEBUG(0,("\nRead from pipe failed (%s)", strerror(errno)));
|
|
}
|
|
|
|
/* if we've cancelled the op, there is no req, so just clean up. */
|
|
if (state->cancelled == true) {
|
|
TALLOC_FREE(state);
|
|
return;
|
|
}
|
|
|
|
req = state->req;
|
|
|
|
if (req) {
|
|
tevent_req_done(req);
|
|
}
|
|
return;
|
|
}
|
|
|
|
static bool init_gluster_aio(struct vfs_handle_struct *handle)
|
|
{
|
|
int fds[2];
|
|
int ret = -1;
|
|
|
|
if (read_fd != -1) {
|
|
/*
|
|
* Already initialized.
|
|
*/
|
|
return true;
|
|
}
|
|
|
|
ret = pipe(fds);
|
|
if (ret == -1) {
|
|
goto fail;
|
|
}
|
|
|
|
read_fd = fds[0];
|
|
write_fd = fds[1];
|
|
|
|
aio_read_event = tevent_add_fd(handle->conn->sconn->ev_ctx,
|
|
NULL,
|
|
read_fd,
|
|
TEVENT_FD_READ,
|
|
aio_tevent_fd_done,
|
|
NULL);
|
|
if (aio_read_event == NULL) {
|
|
goto fail;
|
|
}
|
|
|
|
return true;
|
|
fail:
|
|
TALLOC_FREE(aio_read_event);
|
|
if (read_fd != -1) {
|
|
close(read_fd);
|
|
close(write_fd);
|
|
read_fd = -1;
|
|
write_fd = -1;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static struct glusterfs_aio_state *aio_state_create(TALLOC_CTX *mem_ctx)
|
|
{
|
|
struct tevent_req *req = NULL;
|
|
struct glusterfs_aio_state *state = NULL;
|
|
struct glusterfs_aio_wrapper *wrapper = NULL;
|
|
|
|
req = tevent_req_create(mem_ctx, &wrapper, struct glusterfs_aio_wrapper);
|
|
|
|
if (req == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
state = talloc_zero(NULL, struct glusterfs_aio_state);
|
|
|
|
if (state == NULL) {
|
|
TALLOC_FREE(req);
|
|
return NULL;
|
|
}
|
|
|
|
talloc_set_destructor(wrapper, aio_wrapper_destructor);
|
|
state->cancelled = false;
|
|
state->req = req;
|
|
|
|
wrapper->state = state;
|
|
|
|
return state;
|
|
}
|
|
|
|
static struct tevent_req *vfs_gluster_pread_send(struct vfs_handle_struct
|
|
*handle, TALLOC_CTX *mem_ctx,
|
|
struct tevent_context *ev,
|
|
files_struct *fsp,
|
|
void *data, size_t n,
|
|
off_t offset)
|
|
{
|
|
struct glusterfs_aio_state *state = NULL;
|
|
struct tevent_req *req = NULL;
|
|
int ret = 0;
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return NULL;
|
|
}
|
|
|
|
state = aio_state_create(mem_ctx);
|
|
|
|
if (state == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
req = state->req;
|
|
|
|
if (!init_gluster_aio(handle)) {
|
|
tevent_req_error(req, EIO);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
/*
|
|
* aio_glusterfs_done and aio_tevent_fd_done()
|
|
* use the raw tevent context. We need to use
|
|
* tevent_req_defer_callback() in order to
|
|
* use the event context we're started with.
|
|
*/
|
|
tevent_req_defer_callback(req, ev);
|
|
|
|
PROFILE_TIMESTAMP(&state->start);
|
|
ret = glfs_pread_async(glfd, data, n, offset, 0, aio_glusterfs_done,
|
|
state);
|
|
if (ret < 0) {
|
|
tevent_req_error(req, -ret);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
return req;
|
|
}
|
|
|
|
static struct tevent_req *vfs_gluster_pwrite_send(struct vfs_handle_struct
|
|
*handle, TALLOC_CTX *mem_ctx,
|
|
struct tevent_context *ev,
|
|
files_struct *fsp,
|
|
const void *data, size_t n,
|
|
off_t offset)
|
|
{
|
|
struct glusterfs_aio_state *state = NULL;
|
|
struct tevent_req *req = NULL;
|
|
int ret = 0;
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return NULL;
|
|
}
|
|
|
|
state = aio_state_create(mem_ctx);
|
|
|
|
if (state == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
req = state->req;
|
|
|
|
if (!init_gluster_aio(handle)) {
|
|
tevent_req_error(req, EIO);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
/*
|
|
* aio_glusterfs_done and aio_tevent_fd_done()
|
|
* use the raw tevent context. We need to use
|
|
* tevent_req_defer_callback() in order to
|
|
* use the event context we're started with.
|
|
*/
|
|
tevent_req_defer_callback(req, ev);
|
|
|
|
PROFILE_TIMESTAMP(&state->start);
|
|
ret = glfs_pwrite_async(glfd, data, n, offset, 0, aio_glusterfs_done,
|
|
state);
|
|
if (ret < 0) {
|
|
tevent_req_error(req, -ret);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
return req;
|
|
}
|
|
|
|
static ssize_t vfs_gluster_recv(struct tevent_req *req,
|
|
struct vfs_aio_state *vfs_aio_state)
|
|
{
|
|
struct glusterfs_aio_wrapper *wrapper = NULL;
|
|
int ret = 0;
|
|
|
|
wrapper = tevent_req_data(req, struct glusterfs_aio_wrapper);
|
|
|
|
if (wrapper == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
if (wrapper->state == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
if (tevent_req_is_unix_error(req, &vfs_aio_state->error)) {
|
|
return -1;
|
|
}
|
|
|
|
*vfs_aio_state = wrapper->state->vfs_aio_state;
|
|
ret = wrapper->state->ret;
|
|
|
|
/* Clean up the state, it is in a NULL context. */
|
|
|
|
TALLOC_FREE(wrapper->state);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t vfs_gluster_pwrite(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, const void *data,
|
|
size_t n, off_t offset)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
#ifdef HAVE_GFAPI_VER_7_6
|
|
return glfs_pwrite(glfd, data, n, offset, 0, NULL, NULL);
|
|
#else
|
|
return glfs_pwrite(glfd, data, n, offset, 0);
|
|
#endif
|
|
}
|
|
|
|
static off_t vfs_gluster_lseek(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, off_t offset, int whence)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
return glfs_lseek(glfd, offset, whence);
|
|
}
|
|
|
|
static ssize_t vfs_gluster_sendfile(struct vfs_handle_struct *handle, int tofd,
|
|
files_struct *fromfsp,
|
|
const DATA_BLOB *hdr,
|
|
off_t offset, size_t n)
|
|
{
|
|
errno = ENOTSUP;
|
|
return -1;
|
|
}
|
|
|
|
static ssize_t vfs_gluster_recvfile(struct vfs_handle_struct *handle,
|
|
int fromfd, files_struct *tofsp,
|
|
off_t offset, size_t n)
|
|
{
|
|
errno = ENOTSUP;
|
|
return -1;
|
|
}
|
|
|
|
static int vfs_gluster_rename(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname_src,
|
|
const struct smb_filename *smb_fname_dst)
|
|
{
|
|
return glfs_rename(handle->data, smb_fname_src->base_name,
|
|
smb_fname_dst->base_name);
|
|
}
|
|
|
|
static struct tevent_req *vfs_gluster_fsync_send(struct vfs_handle_struct
|
|
*handle, TALLOC_CTX *mem_ctx,
|
|
struct tevent_context *ev,
|
|
files_struct *fsp)
|
|
{
|
|
struct tevent_req *req = NULL;
|
|
struct glusterfs_aio_state *state = NULL;
|
|
int ret = 0;
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return NULL;
|
|
}
|
|
|
|
state = aio_state_create(mem_ctx);
|
|
|
|
if (state == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
req = state->req;
|
|
|
|
if (!init_gluster_aio(handle)) {
|
|
tevent_req_error(req, EIO);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
/*
|
|
* aio_glusterfs_done and aio_tevent_fd_done()
|
|
* use the raw tevent context. We need to use
|
|
* tevent_req_defer_callback() in order to
|
|
* use the event context we're started with.
|
|
*/
|
|
tevent_req_defer_callback(req, ev);
|
|
|
|
PROFILE_TIMESTAMP(&state->start);
|
|
ret = glfs_fsync_async(glfd, aio_glusterfs_done, state);
|
|
if (ret < 0) {
|
|
tevent_req_error(req, -ret);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
return req;
|
|
}
|
|
|
|
static int vfs_gluster_fsync_recv(struct tevent_req *req,
|
|
struct vfs_aio_state *vfs_aio_state)
|
|
{
|
|
/*
|
|
* Use implicit conversion ssize_t->int
|
|
*/
|
|
return vfs_gluster_recv(req, vfs_aio_state);
|
|
}
|
|
|
|
static int vfs_gluster_stat(struct vfs_handle_struct *handle,
|
|
struct smb_filename *smb_fname)
|
|
{
|
|
struct stat st;
|
|
int ret;
|
|
|
|
ret = glfs_stat(handle->data, smb_fname->base_name, &st);
|
|
if (ret == 0) {
|
|
smb_stat_ex_from_stat(&smb_fname->st, &st);
|
|
}
|
|
if (ret < 0 && errno != ENOENT) {
|
|
DEBUG(0, ("glfs_stat(%s) failed: %s\n",
|
|
smb_fname->base_name, strerror(errno)));
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static int vfs_gluster_fstat(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, SMB_STRUCT_STAT *sbuf)
|
|
{
|
|
struct stat st;
|
|
int ret;
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
ret = glfs_fstat(glfd, &st);
|
|
if (ret == 0) {
|
|
smb_stat_ex_from_stat(sbuf, &st);
|
|
}
|
|
if (ret < 0) {
|
|
DEBUG(0, ("glfs_fstat(%d) failed: %s\n",
|
|
fsp->fh->fd, strerror(errno)));
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static int vfs_gluster_lstat(struct vfs_handle_struct *handle,
|
|
struct smb_filename *smb_fname)
|
|
{
|
|
struct stat st;
|
|
int ret;
|
|
|
|
ret = glfs_lstat(handle->data, smb_fname->base_name, &st);
|
|
if (ret == 0) {
|
|
smb_stat_ex_from_stat(&smb_fname->st, &st);
|
|
}
|
|
if (ret < 0 && errno != ENOENT) {
|
|
DEBUG(0, ("glfs_lstat(%s) failed: %s\n",
|
|
smb_fname->base_name, strerror(errno)));
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static uint64_t vfs_gluster_get_alloc_size(struct vfs_handle_struct *handle,
|
|
files_struct *fsp,
|
|
const SMB_STRUCT_STAT *sbuf)
|
|
{
|
|
return sbuf->st_ex_blocks * 512;
|
|
}
|
|
|
|
static int vfs_gluster_unlink(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname)
|
|
{
|
|
return glfs_unlink(handle->data, smb_fname->base_name);
|
|
}
|
|
|
|
static int vfs_gluster_chmod(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
mode_t mode)
|
|
{
|
|
return glfs_chmod(handle->data, smb_fname->base_name, mode);
|
|
}
|
|
|
|
static int vfs_gluster_fchmod(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, mode_t mode)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
return glfs_fchmod(glfd, mode);
|
|
}
|
|
|
|
static int vfs_gluster_chown(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
uid_t uid,
|
|
gid_t gid)
|
|
{
|
|
return glfs_chown(handle->data, smb_fname->base_name, uid, gid);
|
|
}
|
|
|
|
static int vfs_gluster_fchown(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, uid_t uid, gid_t gid)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
return glfs_fchown(glfd, uid, gid);
|
|
}
|
|
|
|
static int vfs_gluster_lchown(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
uid_t uid,
|
|
gid_t gid)
|
|
{
|
|
return glfs_lchown(handle->data, smb_fname->base_name, uid, gid);
|
|
}
|
|
|
|
static int vfs_gluster_chdir(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname)
|
|
{
|
|
return glfs_chdir(handle->data, smb_fname->base_name);
|
|
}
|
|
|
|
static struct smb_filename *vfs_gluster_getwd(struct vfs_handle_struct *handle,
|
|
TALLOC_CTX *ctx)
|
|
{
|
|
char *cwd;
|
|
char *ret;
|
|
struct smb_filename *smb_fname = NULL;
|
|
|
|
cwd = SMB_CALLOC_ARRAY(char, PATH_MAX);
|
|
if (cwd == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
ret = glfs_getcwd(handle->data, cwd, PATH_MAX - 1);
|
|
if (ret == NULL) {
|
|
SAFE_FREE(cwd);
|
|
return NULL;
|
|
}
|
|
smb_fname = synthetic_smb_fname(ctx,
|
|
ret,
|
|
NULL,
|
|
NULL,
|
|
0);
|
|
SAFE_FREE(cwd);
|
|
return smb_fname;
|
|
}
|
|
|
|
static int vfs_gluster_ntimes(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
struct smb_file_time *ft)
|
|
{
|
|
struct timespec times[2];
|
|
|
|
if (null_timespec(ft->atime)) {
|
|
times[0].tv_sec = smb_fname->st.st_ex_atime.tv_sec;
|
|
times[0].tv_nsec = smb_fname->st.st_ex_atime.tv_nsec;
|
|
} else {
|
|
times[0].tv_sec = ft->atime.tv_sec;
|
|
times[0].tv_nsec = ft->atime.tv_nsec;
|
|
}
|
|
|
|
if (null_timespec(ft->mtime)) {
|
|
times[1].tv_sec = smb_fname->st.st_ex_mtime.tv_sec;
|
|
times[1].tv_nsec = smb_fname->st.st_ex_mtime.tv_nsec;
|
|
} else {
|
|
times[1].tv_sec = ft->mtime.tv_sec;
|
|
times[1].tv_nsec = ft->mtime.tv_nsec;
|
|
}
|
|
|
|
if ((timespec_compare(×[0],
|
|
&smb_fname->st.st_ex_atime) == 0) &&
|
|
(timespec_compare(×[1],
|
|
&smb_fname->st.st_ex_mtime) == 0)) {
|
|
return 0;
|
|
}
|
|
|
|
return glfs_utimens(handle->data, smb_fname->base_name, times);
|
|
}
|
|
|
|
static int vfs_gluster_ftruncate(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, off_t offset)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
#ifdef HAVE_GFAPI_VER_7_6
|
|
return glfs_ftruncate(glfd, offset, NULL, NULL);
|
|
#else
|
|
return glfs_ftruncate(glfd, offset);
|
|
#endif
|
|
}
|
|
|
|
static int vfs_gluster_fallocate(struct vfs_handle_struct *handle,
|
|
struct files_struct *fsp,
|
|
uint32_t mode,
|
|
off_t offset, off_t len)
|
|
{
|
|
#ifdef HAVE_GFAPI_VER_6
|
|
int keep_size, punch_hole;
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
keep_size = mode & VFS_FALLOCATE_FL_KEEP_SIZE;
|
|
punch_hole = mode & VFS_FALLOCATE_FL_PUNCH_HOLE;
|
|
|
|
mode &= ~(VFS_FALLOCATE_FL_KEEP_SIZE|VFS_FALLOCATE_FL_PUNCH_HOLE);
|
|
if (mode != 0) {
|
|
errno = ENOTSUP;
|
|
return -1;
|
|
}
|
|
|
|
if (punch_hole) {
|
|
return glfs_discard(glfd, offset, len);
|
|
}
|
|
|
|
return glfs_fallocate(glfd, keep_size, offset, len);
|
|
#else
|
|
errno = ENOTSUP;
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
static struct smb_filename *vfs_gluster_realpath(struct vfs_handle_struct *handle,
|
|
TALLOC_CTX *ctx,
|
|
const struct smb_filename *smb_fname)
|
|
{
|
|
char *result = NULL;
|
|
struct smb_filename *result_fname = NULL;
|
|
char *resolved_path = SMB_MALLOC_ARRAY(char, PATH_MAX+1);
|
|
|
|
if (resolved_path == NULL) {
|
|
errno = ENOMEM;
|
|
return NULL;
|
|
}
|
|
|
|
result = glfs_realpath(handle->data,
|
|
smb_fname->base_name,
|
|
resolved_path);
|
|
if (result != NULL) {
|
|
result_fname = synthetic_smb_fname(ctx, result, NULL, NULL, 0);
|
|
}
|
|
|
|
SAFE_FREE(resolved_path);
|
|
return result_fname;
|
|
}
|
|
|
|
static bool vfs_gluster_lock(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, int op, off_t offset,
|
|
off_t count, int type)
|
|
{
|
|
struct flock flock = { 0, };
|
|
int ret;
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return false;
|
|
}
|
|
|
|
flock.l_type = type;
|
|
flock.l_whence = SEEK_SET;
|
|
flock.l_start = offset;
|
|
flock.l_len = count;
|
|
flock.l_pid = 0;
|
|
|
|
ret = glfs_posix_lock(glfd, op, &flock);
|
|
|
|
if (op == F_GETLK) {
|
|
/* lock query, true if someone else has locked */
|
|
if ((ret != -1) &&
|
|
(flock.l_type != F_UNLCK) &&
|
|
(flock.l_pid != 0) && (flock.l_pid != getpid()))
|
|
return true;
|
|
/* not me */
|
|
return false;
|
|
}
|
|
|
|
if (ret == -1) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static int vfs_gluster_kernel_flock(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, uint32_t share_mode,
|
|
uint32_t access_mask)
|
|
{
|
|
errno = ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
static int vfs_gluster_linux_setlease(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, int leasetype)
|
|
{
|
|
errno = ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
static bool vfs_gluster_getlock(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, off_t *poffset,
|
|
off_t *pcount, int *ptype, pid_t *ppid)
|
|
{
|
|
struct flock flock = { 0, };
|
|
int ret;
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return false;
|
|
}
|
|
|
|
flock.l_type = *ptype;
|
|
flock.l_whence = SEEK_SET;
|
|
flock.l_start = *poffset;
|
|
flock.l_len = *pcount;
|
|
flock.l_pid = 0;
|
|
|
|
ret = glfs_posix_lock(glfd, F_GETLK, &flock);
|
|
|
|
if (ret == -1) {
|
|
return false;
|
|
}
|
|
|
|
*ptype = flock.l_type;
|
|
*poffset = flock.l_start;
|
|
*pcount = flock.l_len;
|
|
*ppid = flock.l_pid;
|
|
|
|
return true;
|
|
}
|
|
|
|
static int vfs_gluster_symlink(struct vfs_handle_struct *handle,
|
|
const char *link_target,
|
|
const struct smb_filename *new_smb_fname)
|
|
{
|
|
return glfs_symlink(handle->data,
|
|
link_target,
|
|
new_smb_fname->base_name);
|
|
}
|
|
|
|
static int vfs_gluster_readlink(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
char *buf,
|
|
size_t bufsiz)
|
|
{
|
|
return glfs_readlink(handle->data, smb_fname->base_name, buf, bufsiz);
|
|
}
|
|
|
|
static int vfs_gluster_link(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *old_smb_fname,
|
|
const struct smb_filename *new_smb_fname)
|
|
{
|
|
return glfs_link(handle->data,
|
|
old_smb_fname->base_name,
|
|
new_smb_fname->base_name);
|
|
}
|
|
|
|
static int vfs_gluster_mknod(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
mode_t mode,
|
|
SMB_DEV_T dev)
|
|
{
|
|
return glfs_mknod(handle->data, smb_fname->base_name, mode, dev);
|
|
}
|
|
|
|
static int vfs_gluster_chflags(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
unsigned int flags)
|
|
{
|
|
errno = ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
static int vfs_gluster_get_real_filename(struct vfs_handle_struct *handle,
|
|
const char *path, const char *name,
|
|
TALLOC_CTX *mem_ctx, char **found_name)
|
|
{
|
|
int ret;
|
|
char key_buf[GLUSTER_NAME_MAX + 64];
|
|
char val_buf[GLUSTER_NAME_MAX + 1];
|
|
|
|
if (strlen(name) >= GLUSTER_NAME_MAX) {
|
|
errno = ENAMETOOLONG;
|
|
return -1;
|
|
}
|
|
|
|
snprintf(key_buf, GLUSTER_NAME_MAX + 64,
|
|
"glusterfs.get_real_filename:%s", name);
|
|
|
|
ret = glfs_getxattr(handle->data, path, key_buf, val_buf,
|
|
GLUSTER_NAME_MAX + 1);
|
|
if (ret == -1) {
|
|
if (errno == ENOATTR) {
|
|
errno = ENOENT;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
*found_name = talloc_strdup(mem_ctx, val_buf);
|
|
if (found_name[0] == NULL) {
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static const char *vfs_gluster_connectpath(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname)
|
|
{
|
|
return handle->conn->connectpath;
|
|
}
|
|
|
|
/* EA Operations */
|
|
|
|
static ssize_t vfs_gluster_getxattr(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
const char *name,
|
|
void *value,
|
|
size_t size)
|
|
{
|
|
return glfs_getxattr(handle->data, smb_fname->base_name,
|
|
name, value, size);
|
|
}
|
|
|
|
static ssize_t vfs_gluster_fgetxattr(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, const char *name,
|
|
void *value, size_t size)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
return glfs_fgetxattr(glfd, name, value, size);
|
|
}
|
|
|
|
static ssize_t vfs_gluster_listxattr(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
char *list,
|
|
size_t size)
|
|
{
|
|
return glfs_listxattr(handle->data, smb_fname->base_name, list, size);
|
|
}
|
|
|
|
static ssize_t vfs_gluster_flistxattr(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, char *list,
|
|
size_t size)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
return glfs_flistxattr(glfd, list, size);
|
|
}
|
|
|
|
static int vfs_gluster_removexattr(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
const char *name)
|
|
{
|
|
return glfs_removexattr(handle->data, smb_fname->base_name, name);
|
|
}
|
|
|
|
static int vfs_gluster_fremovexattr(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, const char *name)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
return glfs_fremovexattr(glfd, name);
|
|
}
|
|
|
|
static int vfs_gluster_setxattr(struct vfs_handle_struct *handle,
|
|
const struct smb_filename *smb_fname,
|
|
const char *name,
|
|
const void *value, size_t size, int flags)
|
|
{
|
|
return glfs_setxattr(handle->data, smb_fname->base_name, name, value, size, flags);
|
|
}
|
|
|
|
static int vfs_gluster_fsetxattr(struct vfs_handle_struct *handle,
|
|
files_struct *fsp, const char *name,
|
|
const void *value, size_t size, int flags)
|
|
{
|
|
glfs_fd_t *glfd = vfs_gluster_fetch_glfd(handle, fsp);
|
|
if (glfd == NULL) {
|
|
DBG_ERR("Failed to fetch gluster fd\n");
|
|
return -1;
|
|
}
|
|
|
|
return glfs_fsetxattr(glfd, name, value, size, flags);
|
|
}
|
|
|
|
/* AIO Operations */
|
|
|
|
static bool vfs_gluster_aio_force(struct vfs_handle_struct *handle,
|
|
files_struct *fsp)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
static struct vfs_fn_pointers glusterfs_fns = {
|
|
|
|
/* Disk Operations */
|
|
|
|
.connect_fn = vfs_gluster_connect,
|
|
.disconnect_fn = vfs_gluster_disconnect,
|
|
.disk_free_fn = vfs_gluster_disk_free,
|
|
.get_quota_fn = vfs_gluster_get_quota,
|
|
.set_quota_fn = vfs_gluster_set_quota,
|
|
.statvfs_fn = vfs_gluster_statvfs,
|
|
.fs_capabilities_fn = vfs_gluster_fs_capabilities,
|
|
|
|
.get_dfs_referrals_fn = NULL,
|
|
|
|
/* Directory Operations */
|
|
|
|
.opendir_fn = vfs_gluster_opendir,
|
|
.fdopendir_fn = vfs_gluster_fdopendir,
|
|
.readdir_fn = vfs_gluster_readdir,
|
|
.seekdir_fn = vfs_gluster_seekdir,
|
|
.telldir_fn = vfs_gluster_telldir,
|
|
.rewind_dir_fn = vfs_gluster_rewinddir,
|
|
.mkdir_fn = vfs_gluster_mkdir,
|
|
.rmdir_fn = vfs_gluster_rmdir,
|
|
.closedir_fn = vfs_gluster_closedir,
|
|
|
|
/* File Operations */
|
|
|
|
.open_fn = vfs_gluster_open,
|
|
.create_file_fn = NULL,
|
|
.close_fn = vfs_gluster_close,
|
|
.pread_fn = vfs_gluster_pread,
|
|
.pread_send_fn = vfs_gluster_pread_send,
|
|
.pread_recv_fn = vfs_gluster_recv,
|
|
.pwrite_fn = vfs_gluster_pwrite,
|
|
.pwrite_send_fn = vfs_gluster_pwrite_send,
|
|
.pwrite_recv_fn = vfs_gluster_recv,
|
|
.lseek_fn = vfs_gluster_lseek,
|
|
.sendfile_fn = vfs_gluster_sendfile,
|
|
.recvfile_fn = vfs_gluster_recvfile,
|
|
.rename_fn = vfs_gluster_rename,
|
|
.fsync_send_fn = vfs_gluster_fsync_send,
|
|
.fsync_recv_fn = vfs_gluster_fsync_recv,
|
|
|
|
.stat_fn = vfs_gluster_stat,
|
|
.fstat_fn = vfs_gluster_fstat,
|
|
.lstat_fn = vfs_gluster_lstat,
|
|
.get_alloc_size_fn = vfs_gluster_get_alloc_size,
|
|
.unlink_fn = vfs_gluster_unlink,
|
|
|
|
.chmod_fn = vfs_gluster_chmod,
|
|
.fchmod_fn = vfs_gluster_fchmod,
|
|
.chown_fn = vfs_gluster_chown,
|
|
.fchown_fn = vfs_gluster_fchown,
|
|
.lchown_fn = vfs_gluster_lchown,
|
|
.chdir_fn = vfs_gluster_chdir,
|
|
.getwd_fn = vfs_gluster_getwd,
|
|
.ntimes_fn = vfs_gluster_ntimes,
|
|
.ftruncate_fn = vfs_gluster_ftruncate,
|
|
.fallocate_fn = vfs_gluster_fallocate,
|
|
.lock_fn = vfs_gluster_lock,
|
|
.kernel_flock_fn = vfs_gluster_kernel_flock,
|
|
.linux_setlease_fn = vfs_gluster_linux_setlease,
|
|
.getlock_fn = vfs_gluster_getlock,
|
|
.symlink_fn = vfs_gluster_symlink,
|
|
.readlink_fn = vfs_gluster_readlink,
|
|
.link_fn = vfs_gluster_link,
|
|
.mknod_fn = vfs_gluster_mknod,
|
|
.realpath_fn = vfs_gluster_realpath,
|
|
.chflags_fn = vfs_gluster_chflags,
|
|
.file_id_create_fn = NULL,
|
|
.streaminfo_fn = NULL,
|
|
.get_real_filename_fn = vfs_gluster_get_real_filename,
|
|
.connectpath_fn = vfs_gluster_connectpath,
|
|
|
|
.brl_lock_windows_fn = NULL,
|
|
.brl_unlock_windows_fn = NULL,
|
|
.strict_lock_check_fn = NULL,
|
|
.translate_name_fn = NULL,
|
|
.fsctl_fn = NULL,
|
|
|
|
/* NT ACL Operations */
|
|
.fget_nt_acl_fn = NULL,
|
|
.get_nt_acl_fn = NULL,
|
|
.fset_nt_acl_fn = NULL,
|
|
.audit_file_fn = NULL,
|
|
|
|
/* Posix ACL Operations */
|
|
.sys_acl_get_file_fn = posixacl_xattr_acl_get_file,
|
|
.sys_acl_get_fd_fn = posixacl_xattr_acl_get_fd,
|
|
.sys_acl_blob_get_file_fn = posix_sys_acl_blob_get_file,
|
|
.sys_acl_blob_get_fd_fn = posix_sys_acl_blob_get_fd,
|
|
.sys_acl_set_file_fn = posixacl_xattr_acl_set_file,
|
|
.sys_acl_set_fd_fn = posixacl_xattr_acl_set_fd,
|
|
.sys_acl_delete_def_file_fn = posixacl_xattr_acl_delete_def_file,
|
|
|
|
/* EA Operations */
|
|
.getxattr_fn = vfs_gluster_getxattr,
|
|
.getxattrat_send_fn = vfs_not_implemented_getxattrat_send,
|
|
.getxattrat_recv_fn = vfs_not_implemented_getxattrat_recv,
|
|
.fgetxattr_fn = vfs_gluster_fgetxattr,
|
|
.listxattr_fn = vfs_gluster_listxattr,
|
|
.flistxattr_fn = vfs_gluster_flistxattr,
|
|
.removexattr_fn = vfs_gluster_removexattr,
|
|
.fremovexattr_fn = vfs_gluster_fremovexattr,
|
|
.setxattr_fn = vfs_gluster_setxattr,
|
|
.fsetxattr_fn = vfs_gluster_fsetxattr,
|
|
|
|
/* AIO Operations */
|
|
.aio_force_fn = vfs_gluster_aio_force,
|
|
|
|
/* Durable handle Operations */
|
|
.durable_cookie_fn = NULL,
|
|
.durable_disconnect_fn = NULL,
|
|
.durable_reconnect_fn = NULL,
|
|
};
|
|
|
|
static_decl_vfs;
|
|
NTSTATUS vfs_glusterfs_init(TALLOC_CTX *ctx)
|
|
{
|
|
return smb_register_vfs(SMB_VFS_INTERFACE_VERSION,
|
|
"glusterfs", &glusterfs_fns);
|
|
}
|