summaryrefslogtreecommitdiff
path: root/src/cow-extent-map.c
blob: 6e7ee570e53f52f621d0c302c269180bcf943c20 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
#define _GNU_SOURCE   /* for program_invocation_name */
#include <assert.h>   /* for assert(3p) */
#include <errno.h>    /* for errno, program_invocation_name */
#include <error.h>    /* for error(3gnu) */
#include <fcntl.h>    /* for open(2) and O_RDONLY */
#include <getopt.h>   /* for getopt_long(3gnu), struct option, optind, optarg */
#include <inttypes.h> /* PRI* */
#include <stdbool.h>  /* for bool, true, false */
#include <stdio.h>    /* for printf(3p), fprintf(3p), stderr */
#include <stdlib.h>   /* free(3p), exit(3p), EXIT_SUCCESS, EXIT_FAILURE */
#include <string.h>   /* for strcmp(3p) */

#include "extent-map.h" /* for uint32_t, fiemap stuff */ 

#define EXIT_INVALIDARGUMENT 2
#define errusage(format, ...) do { \
	error(0, 0, format, ## __VA_ARGS__); \
	fprintf(stderr, "Try '%s --help' for more information.\n", program_invocation_name); \
	exit(EXIT_INVALIDARGUMENT); \
} while(0)

bool indent;

int print_extent(struct fiemap_extent extent) {
	char *flagstr = strextentflags(extent.fe_flags);
	printf("%s"
	       "logical=%"PRIu64" "
	       "physical=%"PRIu64" "
	       "length=%"PRIu64" "
	       "flags=0x%08"PRIx32" (%s)\n",
	       indent ? "\t" : "",
	       extent.fe_logical,
	       extent.fe_physical,
	       extent.fe_length,
	       extent.fe_flags,
	       flagstr);
	free(flagstr);
	return 0;
}

void usage() {
	printf("Usage: %s [OPTIONS] FILENAME...\n"
	       "Print a file extent map.\n"
	       "\n"
	       "OPTIONS:\n"
	       "  --stream=data   Map extents in the data stream (default).\n"
	       "  --stream=xattr  Map extents in the extended attribute stream.\n"
	       "  --sync          Sync the file before mapping extents (default).\n"
	       "  --no-sync       Don't sync the file before mapping extents.\n"
	       "  -m, --machine   Display the output in a more machine-friendly format.\n"
	       "  -h, --help      Display this help and exit.\n",
	       program_invocation_name);
}

int main(int argc, char *argv[]) {
	char *stream = "data";
	bool machine = false;
	int sync = 1;
	struct option long_options[] = {
		{"stream",  optional_argument, NULL, 1},
		{"sync",    no_argument,       &sync, 1},
		{"no-sync", no_argument,       &sync, 0},
		{"help",    no_argument,       NULL, 'h'},
		{0},
	};
	int flag;
	while ((flag = getopt_long(argc, argv, "mh", long_options, NULL)) >= 0) {
		switch (flag) {
		case 0:
			break;
		case 1:
			stream = optarg;
			break;
		case 'm':
			machine = true;
			break;
		case 'h':
			usage();
			return EXIT_SUCCESS;
		case '?':
			fprintf(stderr, "Try '%s --help' for more information.\n", program_invocation_name);
			return EXIT_INVALIDARGUMENT;
		default:
			assert(0);
		}
	}
	if (argc - optind < 1)
		errusage("wrong number of arguments");

	uint32_t flags = sync ? FIEMAP_FLAG_SYNC : 0;
	if (strcmp(stream, "data") == 0) {
		/* do nothing */
	} else if (strcmp(stream, "xattr") == 0) {
		flags |= FIEMAP_FLAG_XATTR;
	} else {
		errusage("invalid stream name");
	}

	indent = (argc - optind) > 1 && !machine;

	int status = 0;
	for (int i = optind; i < argc; i++) {
		char *filename = argv[i];
		if (machine) {
			printf("%s", filename);
			putchar('\0');
		} else if (argc - optind > 1)
			printf("%s:\n", filename);
		int fd = open(filename, O_RDONLY);
		if (fd < 0) {
			error(0, errno, "%s", filename);
			status |= EXIT_FAILURE;
			continue;
		}

		int r = fiemap(fd, flags, print_extent);
		if (r < 0) {
			error(0, -r, "%s: FS_IOC_FIEMAP", filename);
			status |= EXIT_FAILURE;
		} else {
			status |= r;
		}
		if (machine)
			putchar('\0');
	}
	return status;
}