summaryrefslogtreecommitdiffstats
path: root/tune/tune.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--tune/tune.c149
1 files changed, 149 insertions, 0 deletions
diff --git a/tune/tune.c b/tune/tune.c
new file mode 100644
index 0000000..135f624
--- /dev/null
+++ b/tune/tune.c
@@ -0,0 +1,149 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Copyright (C) 2019 Namjae Jeon <linkinjeon@kernel.org>
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+#include <getopt.h>
+#include <errno.h>
+#include <locale.h>
+
+#include "exfat_ondisk.h"
+#include "libexfat.h"
+#include "exfat_fs.h"
+
+static void usage(void)
+{
+ fprintf(stderr, "Usage: tune.exfat\n");
+ fprintf(stderr, "\t-l | --print-label Print volume label\n");
+ fprintf(stderr, "\t-L | --set-label=label Set volume label\n");
+ fprintf(stderr, "\t-i | --print-serial Print volume serial\n");
+ fprintf(stderr, "\t-I | --set-serial=value Set volume serial\n");
+ fprintf(stderr, "\t-V | --version Show version\n");
+ fprintf(stderr, "\t-v | --verbose Print debug\n");
+ fprintf(stderr, "\t-h | --help Show help\n");
+
+ exit(EXIT_FAILURE);
+}
+
+static struct option opts[] = {
+ {"print-label", no_argument, NULL, 'l' },
+ {"set-label", required_argument, NULL, 'L' },
+ {"print-serial", no_argument, NULL, 'i' },
+ {"set-serial", required_argument, NULL, 'I' },
+ {"version", no_argument, NULL, 'V' },
+ {"verbose", no_argument, NULL, 'v' },
+ {"help", no_argument, NULL, 'h' },
+ {"?", no_argument, NULL, '?' },
+ {NULL, 0, NULL, 0 }
+};
+
+int main(int argc, char *argv[])
+{
+ int c;
+ int ret = EXIT_FAILURE;
+ struct exfat_blk_dev bd;
+ struct exfat_user_input ui;
+ bool version_only = false;
+ int flags = 0;
+ char label_input[VOLUME_LABEL_BUFFER_SIZE];
+ struct exfat *exfat = NULL;
+ struct pbr *bs;
+
+ init_user_input(&ui);
+
+ if (!setlocale(LC_CTYPE, ""))
+ exfat_err("failed to init locale/codeset\n");
+
+ opterr = 0;
+ while ((c = getopt_long(argc, argv, "I:iL:lVvh", opts, NULL)) != EOF)
+ switch (c) {
+ case 'l':
+ flags = EXFAT_GET_VOLUME_LABEL;
+ break;
+ case 'L':
+ snprintf(label_input, sizeof(label_input), "%s",
+ optarg);
+ flags = EXFAT_SET_VOLUME_LABEL;
+ break;
+ case 'i':
+ flags = EXFAT_GET_VOLUME_SERIAL;
+ break;
+ case 'I':
+ ui.volume_serial = strtoul(optarg, NULL, 0);
+ flags = EXFAT_SET_VOLUME_SERIAL;
+ break;
+ case 'V':
+ version_only = true;
+ break;
+ case 'v':
+ print_level = EXFAT_DEBUG;
+ break;
+ case '?':
+ case 'h':
+ default:
+ usage();
+ }
+
+ show_version();
+ if (version_only)
+ exit(EXIT_FAILURE);
+
+ if (argc < 3)
+ usage();
+
+ memset(ui.dev_name, 0, sizeof(ui.dev_name));
+ snprintf(ui.dev_name, sizeof(ui.dev_name), "%s", argv[argc - 1]);
+
+ ret = exfat_get_blk_dev_info(&ui, &bd);
+ if (ret < 0)
+ goto out;
+
+ /* Mode to change or display volume serial */
+ if (flags == EXFAT_GET_VOLUME_SERIAL) {
+ ret = exfat_show_volume_serial(bd.dev_fd);
+ goto close_fd_out;
+ } else if (flags == EXFAT_SET_VOLUME_SERIAL) {
+ ret = exfat_set_volume_serial(&bd, &ui);
+ goto close_fd_out;
+ }
+
+ ret = read_boot_sect(&bd, &bs);
+ if (ret)
+ goto close_fd_out;
+
+ exfat = exfat_alloc_exfat(&bd, bs);
+ if (!exfat) {
+ free(bs);
+ ret = -ENOMEM;
+ goto close_fd_out;
+ }
+
+ exfat->root = exfat_alloc_inode(ATTR_SUBDIR);
+ if (!exfat->root) {
+ ret = -ENOMEM;
+ goto close_fd_out;
+ }
+
+ exfat->root->first_clus = le32_to_cpu(exfat->bs->bsx.root_cluster);
+ if (exfat_root_clus_count(exfat)) {
+ exfat_err("failed to follow the cluster chain of root\n");
+ exfat_free_inode(exfat->root);
+ ret = -EINVAL;
+ goto close_fd_out;
+ }
+
+ if (flags == EXFAT_GET_VOLUME_LABEL)
+ ret = exfat_read_volume_label(exfat);
+ else if (flags == EXFAT_SET_VOLUME_LABEL)
+ ret = exfat_set_volume_label(exfat, label_input);
+close_fd_out:
+ close(bd.dev_fd);
+ if (exfat)
+ exfat_free_exfat(exfat);
+out:
+ return ret;
+}