mirror of https://github.com/trapexit/mergerfs.git
1297 lines
28 KiB
1297 lines
28 KiB
/*
|
|
ISC License
|
|
|
|
Copyright (c) 2021, Antonio SJ Musumeci <trapexit@spawn.link>
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include "fuse_kernel.h"
|
|
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <inttypes.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
static FILE *g_OUTPUT = NULL;
|
|
|
|
#define PARAM(inarg) (((char *)(inarg)) + sizeof(*(inarg)))
|
|
|
|
static
|
|
int
|
|
debug_set_output_null()
|
|
{
|
|
g_OUTPUT = stderr;
|
|
setvbuf(g_OUTPUT,NULL,_IOLBF,0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static
|
|
int
|
|
debug_set_output_filepath(const char *filepath_)
|
|
{
|
|
FILE *tmp;
|
|
|
|
tmp = fopen(filepath_,"a");
|
|
if(tmp == NULL)
|
|
return -errno;
|
|
|
|
g_OUTPUT = tmp;
|
|
setvbuf(g_OUTPUT,NULL,_IOLBF,0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
debug_set_output(const char *filepath_)
|
|
{
|
|
if(filepath_ == NULL)
|
|
return debug_set_output_null();
|
|
|
|
return debug_set_output_filepath(filepath_);
|
|
}
|
|
|
|
static
|
|
__attribute__((constructor))
|
|
void
|
|
debug_constructor(void)
|
|
{
|
|
debug_set_output(NULL);
|
|
}
|
|
|
|
static
|
|
const
|
|
char*
|
|
open_accmode_to_str(const int flags_)
|
|
{
|
|
switch(flags_ & O_ACCMODE)
|
|
{
|
|
case O_RDWR:
|
|
return "O_RDWR";
|
|
case O_RDONLY:
|
|
return "O_RDONLY";
|
|
case O_WRONLY:
|
|
return "O_WRONLY";
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
#define FUSE_WRITE_FLAG_CASE(X) case FUSE_WRITE_##X: return #X
|
|
|
|
static
|
|
const
|
|
char*
|
|
fuse_write_flag_to_str(const uint32_t offset_)
|
|
{
|
|
switch(1 << offset_)
|
|
{
|
|
FUSE_WRITE_FLAG_CASE(CACHE);
|
|
FUSE_WRITE_FLAG_CASE(LOCKOWNER);
|
|
FUSE_WRITE_FLAG_CASE(KILL_PRIV);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#undef FUSE_WRITE_FLAG_CASE
|
|
|
|
#define OPEN_FLAG_CASE(X) case X: return #X
|
|
|
|
static
|
|
const
|
|
char*
|
|
open_flag_to_str(const uint64_t offset_)
|
|
{
|
|
switch(1 << offset_)
|
|
{
|
|
OPEN_FLAG_CASE(O_APPEND);
|
|
OPEN_FLAG_CASE(O_ASYNC);
|
|
OPEN_FLAG_CASE(O_CLOEXEC);
|
|
OPEN_FLAG_CASE(O_CREAT);
|
|
OPEN_FLAG_CASE(O_DIRECT);
|
|
OPEN_FLAG_CASE(O_DIRECTORY);
|
|
#ifdef O_DSYNC
|
|
OPEN_FLAG_CASE(O_DSYNC);
|
|
#endif
|
|
OPEN_FLAG_CASE(O_EXCL);
|
|
#ifdef O_LARGEFILE
|
|
OPEN_FLAG_CASE(O_LARGEFILE);
|
|
#endif
|
|
#ifdef O_NOATIME
|
|
OPEN_FLAG_CASE(O_NOATIME);
|
|
#endif
|
|
OPEN_FLAG_CASE(O_NOCTTY);
|
|
OPEN_FLAG_CASE(O_NOFOLLOW);
|
|
OPEN_FLAG_CASE(O_NONBLOCK);
|
|
#ifdef O_PATH
|
|
OPEN_FLAG_CASE(O_PATH);
|
|
#endif
|
|
OPEN_FLAG_CASE(O_SYNC);
|
|
#ifdef O_TMPFILE
|
|
OPEN_FLAG_CASE(O_TMPFILE);
|
|
#endif
|
|
OPEN_FLAG_CASE(O_TRUNC);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#undef OPEN_FLAG_CASE
|
|
|
|
#define FUSE_INIT_FLAG_CASE(X) case FUSE_##X: return #X
|
|
|
|
static
|
|
const
|
|
char*
|
|
fuse_flag_to_str(const uint64_t offset_)
|
|
{
|
|
switch(1ULL << offset_)
|
|
{
|
|
FUSE_INIT_FLAG_CASE(ASYNC_READ);
|
|
FUSE_INIT_FLAG_CASE(POSIX_LOCKS);
|
|
FUSE_INIT_FLAG_CASE(FILE_OPS);
|
|
FUSE_INIT_FLAG_CASE(ATOMIC_O_TRUNC);
|
|
FUSE_INIT_FLAG_CASE(EXPORT_SUPPORT);
|
|
FUSE_INIT_FLAG_CASE(BIG_WRITES);
|
|
FUSE_INIT_FLAG_CASE(DONT_MASK);
|
|
FUSE_INIT_FLAG_CASE(SPLICE_WRITE);
|
|
FUSE_INIT_FLAG_CASE(SPLICE_MOVE);
|
|
FUSE_INIT_FLAG_CASE(SPLICE_READ);
|
|
FUSE_INIT_FLAG_CASE(FLOCK_LOCKS);
|
|
FUSE_INIT_FLAG_CASE(HAS_IOCTL_DIR);
|
|
FUSE_INIT_FLAG_CASE(AUTO_INVAL_DATA);
|
|
FUSE_INIT_FLAG_CASE(DO_READDIRPLUS);
|
|
FUSE_INIT_FLAG_CASE(READDIRPLUS_AUTO);
|
|
FUSE_INIT_FLAG_CASE(ASYNC_DIO);
|
|
FUSE_INIT_FLAG_CASE(WRITEBACK_CACHE);
|
|
FUSE_INIT_FLAG_CASE(NO_OPEN_SUPPORT);
|
|
FUSE_INIT_FLAG_CASE(PARALLEL_DIROPS);
|
|
FUSE_INIT_FLAG_CASE(HANDLE_KILLPRIV);
|
|
FUSE_INIT_FLAG_CASE(POSIX_ACL);
|
|
FUSE_INIT_FLAG_CASE(ABORT_ERROR);
|
|
FUSE_INIT_FLAG_CASE(MAX_PAGES);
|
|
FUSE_INIT_FLAG_CASE(CACHE_SYMLINKS);
|
|
FUSE_INIT_FLAG_CASE(NO_OPENDIR_SUPPORT);
|
|
FUSE_INIT_FLAG_CASE(EXPLICIT_INVAL_DATA);
|
|
FUSE_INIT_FLAG_CASE(MAP_ALIGNMENT);
|
|
FUSE_INIT_FLAG_CASE(SUBMOUNTS);
|
|
FUSE_INIT_FLAG_CASE(HANDLE_KILLPRIV_V2);
|
|
FUSE_INIT_FLAG_CASE(SETXATTR_EXT);
|
|
FUSE_INIT_FLAG_CASE(INIT_EXT);
|
|
FUSE_INIT_FLAG_CASE(INIT_RESERVED);
|
|
FUSE_INIT_FLAG_CASE(SECURITY_CTX);
|
|
FUSE_INIT_FLAG_CASE(HAS_INODE_DAX);
|
|
FUSE_INIT_FLAG_CASE(CREATE_SUPP_GROUP);
|
|
FUSE_INIT_FLAG_CASE(HAS_EXPIRE_ONLY);
|
|
FUSE_INIT_FLAG_CASE(DIRECT_IO_ALLOW_MMAP);
|
|
FUSE_INIT_FLAG_CASE(PASSTHROUGH);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#undef FUSE_INIT_FLAG_CASE
|
|
|
|
static
|
|
void
|
|
debug_open_flags(const uint32_t flags_)
|
|
{
|
|
fprintf(stderr,"%s, ",open_accmode_to_str(flags_));
|
|
for(int i = 0; i < (sizeof(flags_) * 8); i++)
|
|
{
|
|
const char *str;
|
|
|
|
if(!(flags_ & (1 << i)))
|
|
continue;
|
|
|
|
str = open_flag_to_str(i);
|
|
if(str == NULL)
|
|
continue;
|
|
|
|
fprintf(stderr,"%s, ",str);
|
|
}
|
|
}
|
|
|
|
#define FOPEN_FLAG_CASE(X) case FOPEN_##X: return #X
|
|
|
|
static
|
|
const
|
|
char*
|
|
fuse_fopen_flag_to_str(const uint32_t offset_)
|
|
{
|
|
switch(1 << offset_)
|
|
{
|
|
FOPEN_FLAG_CASE(DIRECT_IO);
|
|
FOPEN_FLAG_CASE(KEEP_CACHE);
|
|
FOPEN_FLAG_CASE(NONSEEKABLE);
|
|
FOPEN_FLAG_CASE(CACHE_DIR);
|
|
FOPEN_FLAG_CASE(STREAM);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#undef FOPEN_FLAG_CASE
|
|
|
|
void
|
|
debug_fuse_open_out(const struct fuse_open_out *arg_)
|
|
{
|
|
fprintf(stderr,
|
|
"fuse_open_out:"
|
|
" fh=0x%"PRIx64";"
|
|
" open_flags=0x%X (",
|
|
arg_->fh,
|
|
arg_->open_flags);
|
|
for(int i = 0; i < (sizeof(arg_->open_flags) * 8); i++)
|
|
{
|
|
const char *str;
|
|
|
|
if(!(arg_->open_flags & (1 << i)))
|
|
continue;
|
|
|
|
str = fuse_fopen_flag_to_str(i);
|
|
if(str == NULL)
|
|
continue;
|
|
|
|
fprintf(stderr,"%s,",str);
|
|
}
|
|
fprintf(stderr,");\n");
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_lookup(const void *arg_)
|
|
{
|
|
const char *name = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_lookup:"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
name);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_getattr_in(const void *arg_)
|
|
{
|
|
const struct fuse_getattr_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_getattr_in:"
|
|
" getattr_flags=0x%08X;"
|
|
" fh=0x%"PRIx64";\n",
|
|
arg->getattr_flags,
|
|
arg->fh);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_setattr_in(const void *arg_)
|
|
{
|
|
const struct fuse_setattr_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_setattr_in:"
|
|
" valid=%u;"
|
|
" fh=0x%"PRIx64";"
|
|
" size=%zu;"
|
|
" lock_owner=%zu;"
|
|
" atime=%zu;"
|
|
" atimensec=%u;"
|
|
" mtime=%zu;"
|
|
" mtimensec=%u;"
|
|
" ctime=%zu;"
|
|
" ctimensec=%u;"
|
|
" mode=%o;"
|
|
" uid=%u;"
|
|
" gid=%u;"
|
|
"\n"
|
|
,
|
|
arg->valid,
|
|
arg->fh,
|
|
arg->size,
|
|
arg->lock_owner,
|
|
arg->atime,
|
|
arg->atimensec,
|
|
arg->mtime,
|
|
arg->mtimensec,
|
|
arg->ctime,
|
|
arg->ctimensec,
|
|
arg->mode,
|
|
arg->uid,
|
|
arg->gid);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_access_in(const void *arg_)
|
|
{
|
|
const struct fuse_access_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_access_in:"
|
|
" mask=0x%08X;"
|
|
"\n"
|
|
,
|
|
arg->mask);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_mknod_in(const void *arg_)
|
|
{
|
|
const struct fuse_mknod_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_mknod_in:"
|
|
" mode=%o;"
|
|
" rdev=0x%08X;"
|
|
" umask=%o;"
|
|
"\n"
|
|
,
|
|
arg->mode,
|
|
arg->rdev,
|
|
arg->umask);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_mkdir_in(const void *arg_)
|
|
{
|
|
const struct fuse_mkdir_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_mkdir_in:"
|
|
" mode=%o;"
|
|
" umask=%o;"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
arg->mode,
|
|
arg->umask,
|
|
PARAM(arg));
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_unlink(const void *arg_)
|
|
{
|
|
const char *name = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_unlink:"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
name);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_rmdir(const void *arg_)
|
|
{
|
|
const char *name = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_mkdir:"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
name);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_symlink(const void *arg_)
|
|
{
|
|
const char *name;
|
|
const char *linkname;
|
|
|
|
name = arg_;
|
|
linkname = (name + (strlen(name) + 1));
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_mkdir:"
|
|
" linkname=%s;"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
linkname,
|
|
name);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_rename_in(const void *arg_)
|
|
{
|
|
const char *oldname;
|
|
const char *newname;
|
|
const struct fuse_rename_in *arg = arg_;
|
|
|
|
oldname = PARAM(arg);
|
|
newname = (oldname + strlen(oldname) + 1);
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_rename_in:"
|
|
" oldname=%s;"
|
|
" newdir=%zu;"
|
|
" newname=%s;"
|
|
"\n"
|
|
,
|
|
oldname,
|
|
arg->newdir,
|
|
newname);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_link_in(const void *arg_)
|
|
{
|
|
const char *name;
|
|
const struct fuse_link_in *arg = arg_;
|
|
|
|
name = PARAM(arg);
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_link_in:"
|
|
" oldnodeid=%zu;"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
arg->oldnodeid,
|
|
name);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_create_in(const void *arg_)
|
|
{
|
|
const char *name;
|
|
const struct fuse_create_in *arg = arg_;
|
|
|
|
name = PARAM(arg);
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_create_in:"
|
|
" mode=%o;"
|
|
" umask=%o;"
|
|
" name=%s;"
|
|
" flags=0x%X (",
|
|
arg->mode,
|
|
arg->umask,
|
|
name,
|
|
arg->flags);
|
|
debug_open_flags(arg->flags);
|
|
fprintf(g_OUTPUT,");\n");
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_open_in(const void *arg_)
|
|
{
|
|
const struct fuse_open_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_open_in:"
|
|
" flags=0x%08X (",
|
|
arg->flags);
|
|
debug_open_flags(arg->flags);
|
|
fprintf(g_OUTPUT,");\n");
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_read_in(const void *arg_)
|
|
{
|
|
const struct fuse_read_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_read_in:"
|
|
" fh=0x%"PRIx64";"
|
|
" offset=%zu;"
|
|
" size=%u;"
|
|
" read_flags=%X;"
|
|
" lock_owner=0x%"PRIx64";"
|
|
" flags=0x%X ("
|
|
,
|
|
arg->fh,
|
|
arg->offset,
|
|
arg->size,
|
|
arg->read_flags,
|
|
arg->lock_owner,
|
|
arg->flags);
|
|
debug_open_flags(arg->flags);
|
|
fprintf(g_OUTPUT,");\n");
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_write_in(const void *arg_)
|
|
{
|
|
const struct fuse_write_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_write_in:"
|
|
" fh=0x%"PRIx64";"
|
|
" offset=%zu;"
|
|
" size=%u;"
|
|
" lock_owner=0x%"PRIx64";"
|
|
" flags=0x%X ("
|
|
,
|
|
arg->fh,
|
|
arg->offset,
|
|
arg->size,
|
|
arg->lock_owner,
|
|
arg->flags);
|
|
debug_open_flags(arg->flags);
|
|
fprintf(g_OUTPUT,
|
|
"); write_flags=0x%X (",
|
|
arg->write_flags);
|
|
for(int i = 0; i < (sizeof(arg->write_flags) * 8); i++)
|
|
{
|
|
const char *str;
|
|
|
|
if(!(arg->write_flags & (1 << i)))
|
|
continue;
|
|
|
|
str = fuse_write_flag_to_str(i);
|
|
if(str == NULL)
|
|
continue;
|
|
|
|
fprintf(g_OUTPUT,"%s,",str);
|
|
}
|
|
fprintf(g_OUTPUT,");\n");
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_flush_in(const void *arg_)
|
|
{
|
|
const struct fuse_flush_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_flush_in:"
|
|
" fh=0x%"PRIx64";"
|
|
" lock_owner=0x%"PRIx64";"
|
|
"\n"
|
|
,
|
|
arg->fh,
|
|
arg->lock_owner);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_release_in(const void *arg_)
|
|
{
|
|
const struct fuse_release_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_release_in:"
|
|
" fh=0x%"PRIx64";"
|
|
" release_flags=0x%X;"
|
|
" lock_owner=0x%"PRIx64";"
|
|
" flags=0x%X ("
|
|
,
|
|
arg->fh,
|
|
arg->release_flags,
|
|
arg->lock_owner,
|
|
arg->flags);
|
|
debug_open_flags(arg->flags);
|
|
fprintf(g_OUTPUT,");\n");
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_fsync_in(const void *arg_)
|
|
{
|
|
const struct fuse_fsync_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_fsync_in:"
|
|
" fh=0x%"PRIx64";"
|
|
" fsync_flags=0x%X;"
|
|
"\n"
|
|
,
|
|
arg->fh,
|
|
arg->fsync_flags);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_setxattr_in(const void *arg_)
|
|
{
|
|
const char *name;
|
|
const char *value;
|
|
const struct fuse_setxattr_in *arg = arg_;
|
|
|
|
name = PARAM(arg);
|
|
value = (name + strlen(name) + 1);
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_setxattr_in:"
|
|
" size=%u;"
|
|
" flags=0x%X;"
|
|
" name=%s;"
|
|
" value=%s;"
|
|
"\n"
|
|
,
|
|
arg->size,
|
|
arg->flags,
|
|
name,
|
|
value);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_getxattr_in(const void *arg_)
|
|
{
|
|
const char *name;
|
|
const struct fuse_getxattr_in *arg = arg_;
|
|
|
|
name = PARAM(arg);
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_getxattr_in:"
|
|
" size=%u;"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
arg->size,
|
|
name);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_listxattr(const void *arg_)
|
|
{
|
|
const struct fuse_getxattr_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_listxattr:"
|
|
" size=%u;"
|
|
"\n"
|
|
,
|
|
arg->size);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_removexattr(const void *arg_)
|
|
{
|
|
const char *name = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_removexattr:"
|
|
" name=%s;"
|
|
"\n"
|
|
,
|
|
name);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_fallocate_in(const void *arg_)
|
|
{
|
|
const struct fuse_fallocate_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_fallocate_in:"
|
|
" fh=0x%"PRIx64";"
|
|
" offset=%zu;"
|
|
" length=%zu;"
|
|
" mode=%o;"
|
|
"\n"
|
|
,
|
|
arg->fh,
|
|
arg->offset,
|
|
arg->length,
|
|
arg->mode);
|
|
}
|
|
|
|
void
|
|
debug_fuse_init_in(const struct fuse_init_in *arg_)
|
|
{
|
|
uint64_t flags;
|
|
|
|
flags = (((uint64_t)arg_->flags) | ((uint64_t)arg_->flags2) << 32);
|
|
fprintf(g_OUTPUT,
|
|
"FUSE_INIT_IN: "
|
|
" major=%u;"
|
|
" minor=%u;"
|
|
" max_readahead=%u;"
|
|
" flags=0x%016lx (",
|
|
arg_->major,
|
|
arg_->minor,
|
|
arg_->max_readahead,
|
|
flags);
|
|
for(uint64_t i = 0; i < (sizeof(flags)*8); i++)
|
|
{
|
|
const char *str;
|
|
|
|
if(!(flags & (1ULL << i)))
|
|
continue;
|
|
|
|
str = fuse_flag_to_str(i);
|
|
if(str == NULL)
|
|
continue;
|
|
|
|
fprintf(g_OUTPUT,"%s, ",str);
|
|
}
|
|
fprintf(g_OUTPUT,")\n");
|
|
}
|
|
|
|
void
|
|
debug_fuse_init_out(const uint64_t unique_,
|
|
const struct fuse_init_out *arg_,
|
|
const uint64_t argsize_)
|
|
{
|
|
uint64_t flags;
|
|
const struct fuse_init_out *arg = arg_;
|
|
|
|
flags = (((uint64_t)arg->flags) | ((uint64_t)arg->flags2) << 32);
|
|
fprintf(g_OUTPUT,
|
|
/* "unique=0x%016"PRIx64";" */
|
|
/* " opcode=RESPONSE;" */
|
|
/* " error=0 (Success);" */
|
|
/* " len=%"PRIu64"; || " */
|
|
"FUSE_INIT_OUT:"
|
|
" major=%u;"
|
|
" minor=%u;"
|
|
" max_readahead=%u;"
|
|
" flags=0x%016lx ("
|
|
,
|
|
/* unique_, */
|
|
/* sizeof(struct fuse_out_header) + argsize_, */
|
|
arg->major,
|
|
arg->minor,
|
|
arg->max_readahead,
|
|
flags);
|
|
|
|
for(uint64_t i = 0; i < (sizeof(flags)*8); i++)
|
|
{
|
|
const char *str;
|
|
|
|
if(!(flags & (1ULL << i)))
|
|
continue;
|
|
|
|
str = fuse_flag_to_str(i);
|
|
if(str == NULL)
|
|
continue;
|
|
|
|
fprintf(g_OUTPUT,"%s, ",str);
|
|
}
|
|
|
|
fprintf(g_OUTPUT,
|
|
"); max_background=%u;"
|
|
" congestion_threshold=%u;"
|
|
" max_write=%u;"
|
|
" time_gran=%u;"
|
|
" max_pages=%u;"
|
|
" map_alignment=%u;"
|
|
" max_stack_depth=%u;"
|
|
"\n",
|
|
arg->max_background,
|
|
arg->congestion_threshold,
|
|
arg->max_write,
|
|
arg->time_gran,
|
|
arg->max_pages,
|
|
arg->map_alignment,
|
|
arg->max_stack_depth);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_attr(const struct fuse_attr *attr_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"attr:"
|
|
" ino=0x%016"PRIx64";"
|
|
" size=%"PRIu64";"
|
|
" blocks=%"PRIu64";"
|
|
" atime=%"PRIu64";"
|
|
" atimensec=%u;"
|
|
" mtime=%"PRIu64";"
|
|
" mtimensec=%u;"
|
|
" ctime=%"PRIu64";"
|
|
" ctimesec=%u;"
|
|
" mode=%o;"
|
|
" nlink=%u;"
|
|
" uid=%u;"
|
|
" gid=%u;"
|
|
" rdev=%u;"
|
|
" blksize=%u;"
|
|
,
|
|
attr_->ino,
|
|
attr_->size,
|
|
attr_->blocks,
|
|
attr_->atime,
|
|
attr_->atimensec,
|
|
attr_->mtime,
|
|
attr_->mtimensec,
|
|
attr_->ctime,
|
|
attr_->ctimensec,
|
|
attr_->mode,
|
|
attr_->nlink,
|
|
attr_->uid,
|
|
attr_->gid,
|
|
attr_->rdev,
|
|
attr_->blksize);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_entry(const struct fuse_entry_out *entry_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
" fuse_entry_out:"
|
|
" nodeid=0x%016"PRIx64";"
|
|
" generation=0x%016"PRIx64";"
|
|
" entry_valid=%"PRIu64";"
|
|
" entry_valid_nsec=%u;"
|
|
" attr_valid=%"PRIu64";"
|
|
" attr_valid_nsec=%u;"
|
|
" ",
|
|
entry_->nodeid,
|
|
entry_->generation,
|
|
entry_->entry_valid,
|
|
entry_->entry_valid_nsec,
|
|
entry_->attr_valid,
|
|
entry_->attr_valid_nsec);
|
|
debug_fuse_attr(&entry_->attr);
|
|
}
|
|
|
|
void
|
|
debug_fuse_entry_out(const uint64_t unique_,
|
|
const struct fuse_entry_out *arg_,
|
|
const uint64_t argsize_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
,
|
|
unique_,
|
|
sizeof(struct fuse_out_header) + argsize_);
|
|
debug_fuse_entry(arg_);
|
|
|
|
}
|
|
|
|
void
|
|
debug_fuse_attr_out(const uint64_t unique_,
|
|
const struct fuse_attr_out *arg_,
|
|
const uint64_t argsize_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
"fuse_attr_out:"
|
|
" attr_valid=%zu;"
|
|
" attr_valid_nsec=%u;"
|
|
" ino=%zu;"
|
|
" size=%zu;"
|
|
" blocks=%zu;"
|
|
" atime=%zu;"
|
|
" atimensec=%u;"
|
|
" mtime=%zu;"
|
|
" mtimensec=%u;"
|
|
" ctime=%zu;"
|
|
" ctimensec=%u;"
|
|
" mode=%o;"
|
|
" nlink=%u;"
|
|
" uid=%u;"
|
|
" gid=%u;"
|
|
" rdev=%u;"
|
|
" blksize=%u;"
|
|
"\n",
|
|
unique_,
|
|
sizeof(struct fuse_out_header) + argsize_,
|
|
arg_->attr_valid,
|
|
arg_->attr_valid_nsec,
|
|
arg_->attr.ino,
|
|
arg_->attr.size,
|
|
arg_->attr.blocks,
|
|
arg_->attr.atime,
|
|
arg_->attr.atimensec,
|
|
arg_->attr.mtime,
|
|
arg_->attr.mtimensec,
|
|
arg_->attr.ctime,
|
|
arg_->attr.ctimensec,
|
|
arg_->attr.mode,
|
|
arg_->attr.nlink,
|
|
arg_->attr.uid,
|
|
arg_->attr.gid,
|
|
arg_->attr.rdev,
|
|
arg_->attr.blksize);
|
|
}
|
|
|
|
static
|
|
void
|
|
debug_fuse_interrupt_in(const void *arg_)
|
|
{
|
|
const struct fuse_interrupt_in *arg = arg_;
|
|
|
|
fprintf(g_OUTPUT,
|
|
"fuse_interrupt_in:"
|
|
" unique=0x%016"PRIx64";"
|
|
"\n"
|
|
,
|
|
arg->unique);
|
|
}
|
|
|
|
static
|
|
const
|
|
char*
|
|
opcode_name(enum fuse_opcode op_)
|
|
{
|
|
static const char *names[] =
|
|
{
|
|
[FUSE_LOOKUP] = "LOOKUP",
|
|
[FUSE_FORGET] = "FORGET",
|
|
[FUSE_GETATTR] = "GETATTR",
|
|
[FUSE_SETATTR] = "SETATTR",
|
|
[FUSE_READLINK] = "READLINK",
|
|
[FUSE_SYMLINK] = "SYMLINK",
|
|
[FUSE_MKNOD] = "MKNOD",
|
|
[FUSE_MKDIR] = "MKDIR",
|
|
[FUSE_UNLINK] = "UNLINK",
|
|
[FUSE_RMDIR] = "RMDIR",
|
|
[FUSE_RENAME] = "RENAME",
|
|
[FUSE_LINK] = "LINK",
|
|
[FUSE_OPEN] = "OPEN",
|
|
[FUSE_READ] = "READ",
|
|
[FUSE_WRITE] = "WRITE",
|
|
[FUSE_STATFS] = "STATFS",
|
|
[FUSE_RELEASE] = "RELEASE",
|
|
[FUSE_FSYNC] = "FSYNC",
|
|
[FUSE_SETXATTR] = "SETXATTR",
|
|
[FUSE_GETXATTR] = "GETXATTR",
|
|
[FUSE_LISTXATTR] = "LISTXATTR",
|
|
[FUSE_REMOVEXATTR] = "REMOVEXATTR",
|
|
[FUSE_FLUSH] = "FLUSH",
|
|
[FUSE_INIT] = "INIT",
|
|
[FUSE_OPENDIR] = "OPENDIR",
|
|
[FUSE_READDIR] = "READDIR",
|
|
[FUSE_RELEASEDIR] = "RELEASEDIR",
|
|
[FUSE_FSYNCDIR] = "FSYNCDIR",
|
|
[FUSE_GETLK] = "GETLK",
|
|
[FUSE_SETLK] = "SETLK",
|
|
[FUSE_SETLKW] = "SETLKW",
|
|
[FUSE_ACCESS] = "ACCESS",
|
|
[FUSE_CREATE] = "CREATE",
|
|
[FUSE_INTERRUPT] = "INTERRUPT",
|
|
[FUSE_BMAP] = "BMAP",
|
|
[FUSE_DESTROY] = "DESTROY",
|
|
[FUSE_IOCTL] = "IOCTL",
|
|
[FUSE_POLL] = "POLL",
|
|
[FUSE_NOTIFY_REPLY] = "NOTIFY_REPLY",
|
|
[FUSE_BATCH_FORGET] = "BATCH_FORGET",
|
|
[FUSE_FALLOCATE] = "FALLOCATE",
|
|
[FUSE_READDIRPLUS] = "READDIRPLUS",
|
|
[FUSE_RENAME2] = "RENAME2",
|
|
[FUSE_LSEEK] = "LSEEK",
|
|
[FUSE_COPY_FILE_RANGE] = "COPY_FILE_RANGE",
|
|
[FUSE_SETUPMAPPING] = "SETUPMAPPING",
|
|
[FUSE_REMOVEMAPPING] = "REMOVEMAPPING"
|
|
};
|
|
|
|
if(op_ >= (sizeof(names) / sizeof(names[0])))
|
|
return "::UNKNOWN::";
|
|
|
|
return names[op_];
|
|
}
|
|
|
|
void
|
|
debug_fuse_in_header(const struct fuse_in_header *hdr_)
|
|
{
|
|
const void *arg = &hdr_[1];
|
|
|
|
fprintf(stderr,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=%s (%u);"
|
|
" nodeid=%zu;"
|
|
" uid=%u;"
|
|
" gid=%u;"
|
|
" pid=%u; || ",
|
|
hdr_->unique,
|
|
opcode_name(hdr_->opcode),
|
|
hdr_->opcode,
|
|
hdr_->nodeid,
|
|
hdr_->uid,
|
|
hdr_->gid,
|
|
hdr_->pid);
|
|
|
|
switch(hdr_->opcode)
|
|
{
|
|
case FUSE_LOOKUP:
|
|
debug_fuse_lookup(arg);
|
|
break;
|
|
case FUSE_INIT:
|
|
debug_fuse_init_in(arg);
|
|
break;
|
|
case FUSE_GETATTR:
|
|
debug_fuse_getattr_in(arg);
|
|
break;
|
|
case FUSE_SETATTR:
|
|
debug_fuse_setattr_in(arg);
|
|
break;
|
|
case FUSE_ACCESS:
|
|
debug_fuse_access_in(arg);
|
|
break;
|
|
case FUSE_MKNOD:
|
|
debug_fuse_mknod_in(arg);
|
|
break;
|
|
case FUSE_MKDIR:
|
|
debug_fuse_mkdir_in(arg);
|
|
break;
|
|
case FUSE_UNLINK:
|
|
debug_fuse_unlink(arg);
|
|
break;
|
|
case FUSE_RMDIR:
|
|
debug_fuse_rmdir(arg);
|
|
break;
|
|
case FUSE_SYMLINK:
|
|
debug_fuse_symlink(arg);
|
|
break;
|
|
case FUSE_RENAME:
|
|
debug_fuse_rename_in(arg);
|
|
break;
|
|
case FUSE_LINK:
|
|
debug_fuse_link_in(arg);
|
|
break;
|
|
case FUSE_CREATE:
|
|
debug_fuse_create_in(arg);
|
|
break;
|
|
case FUSE_OPEN:
|
|
debug_fuse_open_in(arg);
|
|
break;
|
|
case FUSE_OPENDIR:
|
|
debug_fuse_open_in(arg);
|
|
break;
|
|
case FUSE_READ:
|
|
debug_fuse_read_in(arg);
|
|
break;
|
|
case FUSE_READDIR:
|
|
debug_fuse_read_in(arg);
|
|
break;
|
|
case FUSE_READDIRPLUS:
|
|
debug_fuse_read_in(arg);
|
|
break;
|
|
case FUSE_WRITE:
|
|
debug_fuse_write_in(arg);
|
|
break;
|
|
case FUSE_RELEASE:
|
|
debug_fuse_release_in(arg);
|
|
break;
|
|
case FUSE_RELEASEDIR:
|
|
debug_fuse_release_in(arg);
|
|
break;
|
|
case FUSE_FSYNCDIR:
|
|
debug_fuse_fsync_in(arg);
|
|
break;
|
|
case FUSE_GETXATTR:
|
|
debug_fuse_getxattr_in(arg);
|
|
break;
|
|
case FUSE_LISTXATTR:
|
|
debug_fuse_listxattr(arg);
|
|
break;
|
|
case FUSE_SETXATTR:
|
|
debug_fuse_setxattr_in(arg);
|
|
break;
|
|
case FUSE_REMOVEXATTR:
|
|
debug_fuse_removexattr(arg);
|
|
break;
|
|
case FUSE_FALLOCATE:
|
|
debug_fuse_fallocate_in(arg);
|
|
break;
|
|
case FUSE_FLUSH:
|
|
debug_fuse_flush_in(arg);
|
|
break;
|
|
case FUSE_INTERRUPT:
|
|
debug_fuse_interrupt_in(arg);
|
|
break;
|
|
default:
|
|
fprintf(g_OUTPUT,"FIXME\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
debug_fuse_out_header(const struct fuse_out_header *hdr_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=%d (%s);"
|
|
" len=%"PRIu64";"
|
|
,
|
|
hdr_->unique,
|
|
hdr_->error,
|
|
strerror(-hdr_->error),
|
|
sizeof(struct fuse_out_header));
|
|
}
|
|
|
|
void
|
|
debug_fuse_entry_open_out(const uint64_t unique_,
|
|
const struct fuse_entry_out *entry_,
|
|
const struct fuse_open_out *open_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
,
|
|
unique_,
|
|
sizeof(struct fuse_entry_out) + sizeof(struct fuse_open_out));
|
|
debug_fuse_entry(entry_);
|
|
|
|
}
|
|
|
|
void
|
|
debug_fuse_readlink(const uint64_t unique_,
|
|
const char *linkname_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
"readlink: linkname=%s"
|
|
"\n"
|
|
,
|
|
unique_,
|
|
(sizeof(struct fuse_out_header) + strlen(linkname_)),
|
|
linkname_);
|
|
}
|
|
|
|
void
|
|
debug_fuse_write_out(const uint64_t unique_,
|
|
const struct fuse_write_out *arg_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
" fuse_write_out:"
|
|
" size=%u"
|
|
"\n"
|
|
,
|
|
unique_,
|
|
sizeof(struct fuse_write_out),
|
|
arg_->size);
|
|
}
|
|
|
|
void
|
|
debug_fuse_statfs_out(const uint64_t unique_,
|
|
const struct fuse_statfs_out *arg_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
" fuse_statfs_out:"
|
|
" blocks=%"PRIu64";"
|
|
" bfree=%"PRIu64";"
|
|
" bavail=%"PRIu64";"
|
|
" files=%"PRIu64";"
|
|
" ffree=%"PRIu64";"
|
|
" bsize=%u;"
|
|
" namelen=%u;"
|
|
" frsize=%u;"
|
|
"\n"
|
|
,
|
|
unique_,
|
|
sizeof(struct fuse_statfs_out),
|
|
arg_->st.blocks,
|
|
arg_->st.bfree,
|
|
arg_->st.bavail,
|
|
arg_->st.files,
|
|
arg_->st.ffree,
|
|
arg_->st.bsize,
|
|
arg_->st.namelen,
|
|
arg_->st.frsize);
|
|
}
|
|
|
|
void
|
|
debug_fuse_getxattr_out(const uint64_t unique_,
|
|
const struct fuse_getxattr_out *arg_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
" fuse_getxattr_out:"
|
|
" size=%u;"
|
|
"\n"
|
|
,
|
|
unique_,
|
|
sizeof(struct fuse_out_header) + sizeof(struct fuse_getxattr_out),
|
|
arg_->size);
|
|
|
|
}
|
|
|
|
void
|
|
debug_fuse_lk_out(const uint64_t unique_,
|
|
const struct fuse_lk_out *arg_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
" fuse_file_lock:"
|
|
" start=%"PRIu64";"
|
|
" end=%"PRIu64";"
|
|
" type=%u;"
|
|
" pid=%u;"
|
|
"\n"
|
|
,
|
|
unique_,
|
|
sizeof(struct fuse_out_header) + sizeof(struct fuse_lk_out),
|
|
arg_->lk.start,
|
|
arg_->lk.end,
|
|
arg_->lk.type,
|
|
arg_->lk.pid);
|
|
}
|
|
|
|
void
|
|
debug_fuse_bmap_out(const uint64_t unique_,
|
|
const struct fuse_bmap_out *arg_)
|
|
{
|
|
fprintf(g_OUTPUT,
|
|
"unique=0x%016"PRIx64";"
|
|
" opcode=RESPONSE;"
|
|
" error=0 (Success);"
|
|
" len=%"PRIu64"; || "
|
|
" fuse_bmap_out:"
|
|
" block=%"PRIu64";"
|
|
"\n"
|
|
,
|
|
unique_,
|
|
sizeof(struct fuse_out_header) + sizeof(struct fuse_bmap_out),
|
|
arg_->block);
|
|
|
|
}
|