/*
* Copyright (C) 2013, DataDirect Networks, Inc.
+ *
+ * Copyright (c) 2014, Intel Corporation.
+ *
* Author: Swapnil Pimpale <spimpale@ddn.com>
*/
#include <sys/stat.h>
#include <sys/syscall.h>
-#include <liblustre.h>
-#include <lustre/lustre_user.h>
+#include <linux/lustre/lustre_user.h>
#define MAX_HANDLE_SZ 128
-#if !defined(HAVE_FHANDLE_GLIBC_SUPPORT) && defined(HAVE_FHANDLE_SYSCALLS)
+void usage(char *prog)
+{
+ fprintf(stderr, "usage: %s <filepath> <mount2>\n",
+ prog);
+ fprintf(stderr, "the absolute path of a test file on a "
+ "lustre file system is needed.\n");
+ exit(1);
+}
+
+
+#ifndef HAVE_FHANDLE_SYSCALLS
+
+int main(int argc, char **argv)
+{
+ if (argc != 3)
+ usage(argv[0]);
+
+ fprintf(stderr, "HAVE_FHANDLE_SYSCALLS not defined\n");
+ return 0;
+}
+
+#else
+
+#ifndef HAVE_FHANDLE_GLIBC_SUPPORT
/* Because the kernel supports this functions doesn't mean that glibc does.
* Just in case we define what we need */
-struct file_handle
-{
- unsigned int handle_bytes;
+struct file_handle {
+ __u32 handle_bytes;
int handle_type;
- /* File identifier. */
+ /* file identifier */
unsigned char f_handle[0];
};
+#if defined(_ASM_X86_UNISTD_64_H)
+
+#ifndef __NR_name_to_handle_at
+#define __NR_name_to_handle_at 303
+#endif
+
+#ifndef __NR_open_by_handle_at
+#define __NR_open_by_handle_at 304
+#endif
+
+#elif defined(_ASM_X86_UNISTD_32_H)
+
+#ifndef __NR_name_to_handle_at
+#define __NR_name_to_handle_at 341
+#endif
+
+#ifndef __NR_open_by_handle_at
+#define __NR_open_by_handle_at 342
+#endif
+
+#else
+
#ifndef __NR_name_to_handle_at
#define __NR_name_to_handle_at 264
#endif
#define __NR_open_by_handle_at 265
#endif
+#endif
+
static inline int
name_to_handle_at(int mnt_fd, const char *filename, struct file_handle *fh,
int *mnt_id, int flags)
}
#endif
-void usage(char *prog)
+/* verify a file contents */
+int check_access(const char *filename,
+ int mnt_fd, struct file_handle *fh, struct stat *st_orig)
{
- fprintf(stderr, "usage: %s <filepath>\n",
- prog);
- fprintf(stderr, "the absolute path of a test file on a "
- "lustre file system is needed.\n");
- exit(1);
+ int fd2, rc, len, offset;
+ struct stat st;
+ char *readbuf = NULL;
+
+ /* Open the file handle */
+ fd2 = open_by_handle_at(mnt_fd, fh, O_RDONLY);
+ if (fd2 < 0) {
+ fprintf(stderr, "open_by_handle_at(%s) error: %s\n", filename,
+ strerror(errno));
+ rc = errno;
+ goto out_f_handle;
+ }
+
+ /* Get file size */
+ bzero(&st, sizeof(struct stat));
+ rc = fstat(fd2, &st);
+ if (rc < 0) {
+ fprintf(stderr, "fstat(%s) error: %s\n", filename,
+ strerror(errno));
+ rc = errno;
+ goto out_fd2;
+ }
+
+ /* we can't check a ctime due unlink update */
+ if (st_orig->st_size != st.st_size ||
+ st_orig->st_ino != st.st_ino ||
+ st_orig->st_mtime != st.st_mtime) {
+ fprintf(stderr, "stat data does not match between fopen "
+ "and fhandle case\n");
+ rc = EINVAL;
+ goto out_fd2;
+ }
+
+ if (st.st_size) {
+ len = st.st_blksize;
+ readbuf = malloc(len);
+ if (readbuf == NULL) {
+ fprintf(stderr, "malloc(%d) error: %s\n", len,
+ strerror(errno));
+ rc = errno;
+ goto out_fd2;
+ }
+
+ for (offset = 0; offset < st.st_size; offset += len) {
+ /* read from the file */
+ rc = read(fd2, readbuf, len);
+ if (rc < 0) {
+ fprintf(stderr, "read(%s) error: %s\n",
+ filename, strerror(errno));
+ rc = errno;
+ goto out_readbuf;
+ }
+ }
+ }
+ rc = 0;
+out_readbuf:
+ free(readbuf);
+out_fd2:
+ close(fd2);
+out_f_handle:
+ return rc;
}
int main(int argc, char **argv)
{
-#ifdef HAVE_FHANDLE_SYSCALLS
- char *filename, *file, *mount_point = NULL, *readbuf = NULL;
- int ret, rc = -EINVAL, mnt_id, mnt_fd, fd1, fd2, i, len, offset;
+ char *filename, *file;
+ int ret, rc = -EINVAL, mnt_fd, mnt_id, fd1, i;
struct file_handle *fh = NULL;
- int file_size, mtime, ctime;
struct lu_fid *parent, *fid;
- struct mntent *ent;
struct stat st;
- __ino_t inode;
- FILE *mntpt;
- if (argc != 2)
+ if (argc != 3)
usage(argv[0]);
file = argv[1];
goto out;
}
+ if (*argv[2] != '/') {
+ fprintf(stderr, "Need the absolete path of the mount point\n");
+ goto out;
+ }
+ filename = rindex(file, '/') + 1;
+
fd1 = open(file, O_RDONLY);
if (fd1 < 0) {
fprintf(stderr, "open file %s error: %s\n",
goto out_fd1;
}
- inode = st.st_ino;
- mtime = st.st_mtime;
- ctime = st.st_ctime;
- file_size = st.st_size;
-
- /* Now for the setup to use fhandles */
- mntpt = setmntent("/etc/mtab", "r");
- if (mntpt == NULL) {
- fprintf(stderr, "setmntent error: %s\n",
- strerror(errno));
- rc = errno;
- goto out_fd1;
- }
-
- while (NULL != (ent = getmntent(mntpt))) {
- if ((strncmp(file, ent->mnt_dir, strlen(ent->mnt_dir)) == 0) &&
- (strcmp(ent->mnt_type, "lustre") == 0)) {
- mount_point = ent->mnt_dir;
- break;
- }
- }
- endmntent(mntpt);
-
- if (mount_point == NULL) {
- fprintf(stderr, "file is not located on a lustre file "
- "system?\n");
- goto out_fd1;
- }
-
- filename = rindex(file, '/') + 1;
-
/* Open mount point directory */
- mnt_fd = open(mount_point, O_DIRECTORY);
+ mnt_fd = open(argv[2], O_DIRECTORY);
if (mnt_fd < 0) {
- fprintf(stderr, "open(%s) error: %s\n)", mount_point,
+ fprintf(stderr, "open(%s) error: %s\n)", argv[2],
strerror(errno));
rc = errno;
goto out_fd1;
fh->handle_bytes = MAX_HANDLE_SZ;
/* Convert name to handle */
- ret = name_to_handle_at(mnt_fd, filename, fh, &mnt_id,
+ ret = name_to_handle_at(AT_FDCWD, file, fh, &mnt_id,
AT_SYMLINK_FOLLOW);
if (ret) {
fprintf(stderr, "name_by_handle_at(%s) error: %s\n", filename,
fprintf(stdout, "file's parent FID is "DFID"\n", PFID(parent));
fprintf(stdout, "file FID is "DFID"\n", PFID(fid));
- /* Open the file handle */
- fd2 = open_by_handle_at(mnt_fd, fh, O_RDONLY);
- if (fd2 < 0) {
- fprintf(stderr, "open_by_handle_at(%s) error: %s\n", filename,
- strerror(errno));
- rc = errno;
+ fprintf(stdout, "just access via different mount point - ");
+ rc = check_access(filename, mnt_fd, fh, &st);
+ if (rc != 0)
goto out_f_handle;
- }
+ fprintf(stdout, "OK \n");
- /* Get file size */
- bzero(&st, sizeof(struct stat));
- rc = fstat(fd2, &st);
- if (rc < 0) {
- fprintf(stderr, "fstat(%s) error: %s\n", filename,
- strerror(errno));
- rc = errno;
- goto out_fd2;
- }
-
- if (ctime != st.st_ctime || file_size != st.st_size ||
- inode != st.st_ino || mtime != st.st_mtime) {
- fprintf(stderr, "stat data does not match between fopen "
- "and fhandle case\n");
- goto out_fd2;
+ fprintf(stdout, "access after unlink - ");
+ ret = unlink(file);
+ if (ret < 0) {
+ fprintf(stderr, "can't unlink a file. check permissions?\n");
+ goto out_f_handle;
}
- if (st.st_size) {
- len = st.st_blksize;
- readbuf = malloc(len);
- if (readbuf == NULL) {
- fprintf(stderr, "malloc(%d) error: %s\n", len,
- strerror(errno));
- rc = errno;
- goto out_fd2;
- }
-
- for (offset = 0; offset < st.st_size; offset += len) {
- /* read from the file */
- rc = read(fd2, readbuf, len);
- if (rc < 0) {
- fprintf(stderr, "read(%s) error: %s\n",
- filename, strerror(errno));
- rc = errno;
- goto out_readbuf;
- }
- }
- }
+ rc = check_access(filename, mnt_fd, fh, &st);
+ if (rc != 0)
+ goto out_f_handle;
+ fprintf(stdout, "OK\n");
rc = 0;
fprintf(stdout, "check_fhandle_syscalls test Passed!\n");
-out_readbuf:
- if (readbuf != NULL)
- free(readbuf);
-out_fd2:
- close(fd2);
out_f_handle:
free(fh);
out_mnt_fd:
close(fd1);
out:
return rc;
-#else /* !HAVE_FHANDLE_SYSCALLS */
- if (argc != 2)
- usage(argv[0]);
-
- fprintf(stderr, "HAVE_FHANDLE_SYSCALLS not defined\n");
- return 0;
-#endif /* HAVE_FHANDLE_SYSCALLS */
}
+
+#endif