#include <string.h>
#include <errno.h>
#include <fcntl.h>
+#include <stddef.h>
+#include <assert.h>
-static const char *hello_str = "Hello World!\n";
-static const char *hello_path = "/hello";
+/**
+ * Command line options
+ *
+ * We can't set default values for the char* fields here because
+ * fuse_opt_parse would attempt to free() them when the user specifies
+ * different values on the command line.
+ */
+static struct options {
+ const char *filename;
+ const char *contents;
+ int show_help;
+} options;
+
+#define OPTION(t, p) \
+ { t, offsetof(struct options, p), 1 }
+static const struct fuse_opt option_spec[] = {
+ OPTION("--name=%s", filename),
+ OPTION("--contents=%s", contents),
+ OPTION("-h", show_help),
+ OPTION("--help", show_help),
+ FUSE_OPT_END
+};
static int hello_getattr(const char *path, struct stat *stbuf)
{
if (strcmp(path, "/") == 0) {
stbuf->st_mode = S_IFDIR | 0755;
stbuf->st_nlink = 2;
- } else if (strcmp(path, hello_path) == 0) {
+ } else if (strcmp(path+1, options.filename) == 0) {
stbuf->st_mode = S_IFREG | 0444;
stbuf->st_nlink = 1;
- stbuf->st_size = strlen(hello_str);
+ stbuf->st_size = strlen(options.contents);
} else
res = -ENOENT;
filler(buf, ".", NULL, 0, 0);
filler(buf, "..", NULL, 0, 0);
- filler(buf, hello_path + 1, NULL, 0, 0);
+ filler(buf, options.filename, NULL, 0, 0);
return 0;
}
static int hello_open(const char *path, struct fuse_file_info *fi)
{
- if (strcmp(path, hello_path) != 0)
+ if (strcmp(path+1, options.filename) != 0)
return -ENOENT;
if ((fi->flags & 3) != O_RDONLY)
{
size_t len;
(void) fi;
- if(strcmp(path, hello_path) != 0)
+ if(strcmp(path+1, options.filename) != 0)
return -ENOENT;
- len = strlen(hello_str);
+ len = strlen(options.contents);
if (offset < len) {
if (offset + size > len)
size = len - offset;
- memcpy(buf, hello_str + offset, size);
+ memcpy(buf, options.contents + offset, size);
} else
size = 0;
.read = hello_read,
};
+static void show_help(const char *progname)
+{
+ printf("usage: %s [options] <mountpoint>\n\n", progname);
+ printf("File-system specific options:\n"
+ " --name=<s> Name of the \"hello\" file\n"
+ " (default: \"hello\")\n"
+ " --contents=<s> Contents \"hello\" file\n"
+ " (default \"Hello, World!\\n\")\n"
+ "\n");
+}
+
int main(int argc, char *argv[])
{
- return fuse_main(argc, argv, &hello_oper, NULL);
+ struct fuse_args args = FUSE_ARGS_INIT(argc, argv);
+
+ /* Set defaults -- we have to use strdup so that
+ fuse_opt_parse can free the defaults if other
+ values are specified */
+ options.filename = strdup("hello");
+ options.contents = strdup("Hello World!\n");
+
+ /* Parse options */
+ if (fuse_opt_parse(&args, &options, option_spec, NULL) == -1)
+ return 1;
+
+ /* When --help is specified, first print our own file-system
+ specific help text, then signal fuse_main to show
+ additional help (by adding `--help` to the options again)
+ without usage: line (by setting argv[0] to the empty
+ string) */
+ if (options.show_help) {
+ show_help(argv[0]);
+ assert(fuse_opt_add_arg(&args, "--help") == 0);
+ args.argv[0] = (char*) "";
+ }
+
+ return fuse_main(args.argc, args.argv, &hello_oper, NULL);
}
* main() function.
*
* This function does the following:
- * - parses command line options
+ * - parses command line options, and handles --help and
+ * --version
* - installs signal handlers for INT, HUP, TERM and PIPE
* - registers an exit handler to unmount the filesystem on program exit
* - creates a fuse handle
* - registers the operations
* - calls either the single-threaded or the multi-threaded event loop
*
+ * Most file systems will have to parse some file-system specific
+ * arguments before calling this function. It is recommended to do
+ * this with fuse_opt_parse() and a processing function that passes
+ * through any unknown options (this can also be achieved by just
+ * passing NULL as the processing function). That way, the remaining
+ * options can be passed directly to fuse_main().
+ *
+ * To get a list of the options recognized by fuse_main(), look
+ * at the output when running with ``--help``.
+ *
+ * Normally, fuse_main() includes a basic ``usage: `` message in the
+ * --help output. However, if argv[0] is an empty string, the usage
+ * message is suppressed. This can be used by file systems to print
+ * their own usage line first. See hello.c for an example of how to do
+ * this.
+ *
* Note: this is currently implemented as a macro.
*
* @param argc the argument counter passed to the main() function