1*508ec739SDaniel Rosenberg // SPDX-License-Identifier: GPL-2.0-or-later
2*508ec739SDaniel Rosenberg /*
3*508ec739SDaniel Rosenberg * Copyright (C) 2019 Namjae Jeon <[email protected]>
4*508ec739SDaniel Rosenberg */
5*508ec739SDaniel Rosenberg
6*508ec739SDaniel Rosenberg #include <stdio.h>
7*508ec739SDaniel Rosenberg #include <stdlib.h>
8*508ec739SDaniel Rosenberg #include <string.h>
9*508ec739SDaniel Rosenberg #include <unistd.h>
10*508ec739SDaniel Rosenberg #include <getopt.h>
11*508ec739SDaniel Rosenberg #include <errno.h>
12*508ec739SDaniel Rosenberg #include <locale.h>
13*508ec739SDaniel Rosenberg
14*508ec739SDaniel Rosenberg #include "exfat_ondisk.h"
15*508ec739SDaniel Rosenberg #include "libexfat.h"
16*508ec739SDaniel Rosenberg #include "exfat_fs.h"
17*508ec739SDaniel Rosenberg
usage(void)18*508ec739SDaniel Rosenberg static void usage(void)
19*508ec739SDaniel Rosenberg {
20*508ec739SDaniel Rosenberg fprintf(stderr, "Usage: tune.exfat\n");
21*508ec739SDaniel Rosenberg fprintf(stderr, "\t-l | --print-label Print volume label\n");
22*508ec739SDaniel Rosenberg fprintf(stderr, "\t-L | --set-label=label Set volume label\n");
23*508ec739SDaniel Rosenberg fprintf(stderr, "\t-i | --print-serial Print volume serial\n");
24*508ec739SDaniel Rosenberg fprintf(stderr, "\t-I | --set-serial=value Set volume serial\n");
25*508ec739SDaniel Rosenberg fprintf(stderr, "\t-V | --version Show version\n");
26*508ec739SDaniel Rosenberg fprintf(stderr, "\t-v | --verbose Print debug\n");
27*508ec739SDaniel Rosenberg fprintf(stderr, "\t-h | --help Show help\n");
28*508ec739SDaniel Rosenberg
29*508ec739SDaniel Rosenberg exit(EXIT_FAILURE);
30*508ec739SDaniel Rosenberg }
31*508ec739SDaniel Rosenberg
32*508ec739SDaniel Rosenberg static struct option opts[] = {
33*508ec739SDaniel Rosenberg {"print-label", no_argument, NULL, 'l' },
34*508ec739SDaniel Rosenberg {"set-label", required_argument, NULL, 'L' },
35*508ec739SDaniel Rosenberg {"print-serial", no_argument, NULL, 'i' },
36*508ec739SDaniel Rosenberg {"set-serial", required_argument, NULL, 'I' },
37*508ec739SDaniel Rosenberg {"version", no_argument, NULL, 'V' },
38*508ec739SDaniel Rosenberg {"verbose", no_argument, NULL, 'v' },
39*508ec739SDaniel Rosenberg {"help", no_argument, NULL, 'h' },
40*508ec739SDaniel Rosenberg {"?", no_argument, NULL, '?' },
41*508ec739SDaniel Rosenberg {NULL, 0, NULL, 0 }
42*508ec739SDaniel Rosenberg };
43*508ec739SDaniel Rosenberg
main(int argc,char * argv[])44*508ec739SDaniel Rosenberg int main(int argc, char *argv[])
45*508ec739SDaniel Rosenberg {
46*508ec739SDaniel Rosenberg int c;
47*508ec739SDaniel Rosenberg int ret = EXIT_FAILURE;
48*508ec739SDaniel Rosenberg struct exfat_blk_dev bd;
49*508ec739SDaniel Rosenberg struct exfat_user_input ui;
50*508ec739SDaniel Rosenberg bool version_only = false;
51*508ec739SDaniel Rosenberg int flags = 0;
52*508ec739SDaniel Rosenberg char label_input[VOLUME_LABEL_BUFFER_SIZE];
53*508ec739SDaniel Rosenberg struct exfat *exfat = NULL;
54*508ec739SDaniel Rosenberg struct pbr *bs;
55*508ec739SDaniel Rosenberg
56*508ec739SDaniel Rosenberg init_user_input(&ui);
57*508ec739SDaniel Rosenberg
58*508ec739SDaniel Rosenberg if (!setlocale(LC_CTYPE, ""))
59*508ec739SDaniel Rosenberg exfat_err("failed to init locale/codeset\n");
60*508ec739SDaniel Rosenberg
61*508ec739SDaniel Rosenberg opterr = 0;
62*508ec739SDaniel Rosenberg while ((c = getopt_long(argc, argv, "I:iL:lVvh", opts, NULL)) != EOF)
63*508ec739SDaniel Rosenberg switch (c) {
64*508ec739SDaniel Rosenberg case 'l':
65*508ec739SDaniel Rosenberg flags = EXFAT_GET_VOLUME_LABEL;
66*508ec739SDaniel Rosenberg break;
67*508ec739SDaniel Rosenberg case 'L':
68*508ec739SDaniel Rosenberg snprintf(label_input, sizeof(label_input), "%s",
69*508ec739SDaniel Rosenberg optarg);
70*508ec739SDaniel Rosenberg flags = EXFAT_SET_VOLUME_LABEL;
71*508ec739SDaniel Rosenberg break;
72*508ec739SDaniel Rosenberg case 'i':
73*508ec739SDaniel Rosenberg flags = EXFAT_GET_VOLUME_SERIAL;
74*508ec739SDaniel Rosenberg break;
75*508ec739SDaniel Rosenberg case 'I':
76*508ec739SDaniel Rosenberg ui.volume_serial = strtoul(optarg, NULL, 0);
77*508ec739SDaniel Rosenberg flags = EXFAT_SET_VOLUME_SERIAL;
78*508ec739SDaniel Rosenberg break;
79*508ec739SDaniel Rosenberg case 'V':
80*508ec739SDaniel Rosenberg version_only = true;
81*508ec739SDaniel Rosenberg break;
82*508ec739SDaniel Rosenberg case 'v':
83*508ec739SDaniel Rosenberg print_level = EXFAT_DEBUG;
84*508ec739SDaniel Rosenberg break;
85*508ec739SDaniel Rosenberg case '?':
86*508ec739SDaniel Rosenberg case 'h':
87*508ec739SDaniel Rosenberg default:
88*508ec739SDaniel Rosenberg usage();
89*508ec739SDaniel Rosenberg }
90*508ec739SDaniel Rosenberg
91*508ec739SDaniel Rosenberg show_version();
92*508ec739SDaniel Rosenberg if (version_only)
93*508ec739SDaniel Rosenberg exit(EXIT_FAILURE);
94*508ec739SDaniel Rosenberg
95*508ec739SDaniel Rosenberg if (argc < 3)
96*508ec739SDaniel Rosenberg usage();
97*508ec739SDaniel Rosenberg
98*508ec739SDaniel Rosenberg memset(ui.dev_name, 0, sizeof(ui.dev_name));
99*508ec739SDaniel Rosenberg snprintf(ui.dev_name, sizeof(ui.dev_name), "%s", argv[argc - 1]);
100*508ec739SDaniel Rosenberg
101*508ec739SDaniel Rosenberg ret = exfat_get_blk_dev_info(&ui, &bd);
102*508ec739SDaniel Rosenberg if (ret < 0)
103*508ec739SDaniel Rosenberg goto out;
104*508ec739SDaniel Rosenberg
105*508ec739SDaniel Rosenberg /* Mode to change or display volume serial */
106*508ec739SDaniel Rosenberg if (flags == EXFAT_GET_VOLUME_SERIAL) {
107*508ec739SDaniel Rosenberg ret = exfat_show_volume_serial(bd.dev_fd);
108*508ec739SDaniel Rosenberg goto close_fd_out;
109*508ec739SDaniel Rosenberg } else if (flags == EXFAT_SET_VOLUME_SERIAL) {
110*508ec739SDaniel Rosenberg ret = exfat_set_volume_serial(&bd, &ui);
111*508ec739SDaniel Rosenberg goto close_fd_out;
112*508ec739SDaniel Rosenberg }
113*508ec739SDaniel Rosenberg
114*508ec739SDaniel Rosenberg ret = read_boot_sect(&bd, &bs);
115*508ec739SDaniel Rosenberg if (ret)
116*508ec739SDaniel Rosenberg goto close_fd_out;
117*508ec739SDaniel Rosenberg
118*508ec739SDaniel Rosenberg exfat = exfat_alloc_exfat(&bd, bs);
119*508ec739SDaniel Rosenberg if (!exfat) {
120*508ec739SDaniel Rosenberg free(bs);
121*508ec739SDaniel Rosenberg ret = -ENOMEM;
122*508ec739SDaniel Rosenberg goto close_fd_out;
123*508ec739SDaniel Rosenberg }
124*508ec739SDaniel Rosenberg
125*508ec739SDaniel Rosenberg exfat->root = exfat_alloc_inode(ATTR_SUBDIR);
126*508ec739SDaniel Rosenberg if (!exfat->root) {
127*508ec739SDaniel Rosenberg ret = -ENOMEM;
128*508ec739SDaniel Rosenberg goto close_fd_out;
129*508ec739SDaniel Rosenberg }
130*508ec739SDaniel Rosenberg
131*508ec739SDaniel Rosenberg exfat->root->first_clus = le32_to_cpu(exfat->bs->bsx.root_cluster);
132*508ec739SDaniel Rosenberg if (exfat_root_clus_count(exfat)) {
133*508ec739SDaniel Rosenberg exfat_err("failed to follow the cluster chain of root\n");
134*508ec739SDaniel Rosenberg exfat_free_inode(exfat->root);
135*508ec739SDaniel Rosenberg ret = -EINVAL;
136*508ec739SDaniel Rosenberg goto close_fd_out;
137*508ec739SDaniel Rosenberg }
138*508ec739SDaniel Rosenberg
139*508ec739SDaniel Rosenberg if (flags == EXFAT_GET_VOLUME_LABEL)
140*508ec739SDaniel Rosenberg ret = exfat_read_volume_label(exfat);
141*508ec739SDaniel Rosenberg else if (flags == EXFAT_SET_VOLUME_LABEL)
142*508ec739SDaniel Rosenberg ret = exfat_set_volume_label(exfat, label_input);
143*508ec739SDaniel Rosenberg close_fd_out:
144*508ec739SDaniel Rosenberg close(bd.dev_fd);
145*508ec739SDaniel Rosenberg if (exfat)
146*508ec739SDaniel Rosenberg exfat_free_exfat(exfat);
147*508ec739SDaniel Rosenberg out:
148*508ec739SDaniel Rosenberg return ret;
149*508ec739SDaniel Rosenberg }
150