1987da915Sopenharmony_ci/**
2987da915Sopenharmony_ci * ntfscluster - Part of the Linux-NTFS project.
3987da915Sopenharmony_ci *
4987da915Sopenharmony_ci * Copyright (c) 2002-2003 Richard Russon
5987da915Sopenharmony_ci * Copyright (c) 2005 Anton Altaparmakov
6987da915Sopenharmony_ci * Copyright (c) 2005-2006 Szabolcs Szakacsits
7987da915Sopenharmony_ci *
8987da915Sopenharmony_ci * This utility will locate the owner of any given sector or cluster.
9987da915Sopenharmony_ci *
10987da915Sopenharmony_ci * This program is free software; you can redistribute it and/or modify
11987da915Sopenharmony_ci * it under the terms of the GNU General Public License as published by
12987da915Sopenharmony_ci * the Free Software Foundation; either version 2 of the License, or
13987da915Sopenharmony_ci * (at your option) any later version.
14987da915Sopenharmony_ci *
15987da915Sopenharmony_ci * This program is distributed in the hope that it will be useful,
16987da915Sopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
17987da915Sopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18987da915Sopenharmony_ci * GNU General Public License for more details.
19987da915Sopenharmony_ci *
20987da915Sopenharmony_ci * You should have received a copy of the GNU General Public License
21987da915Sopenharmony_ci * along with this program (in the main directory of the Linux-NTFS
22987da915Sopenharmony_ci * distribution in the file COPYING); if not, write to the Free Software
23987da915Sopenharmony_ci * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
24987da915Sopenharmony_ci */
25987da915Sopenharmony_ci
26987da915Sopenharmony_ci#include "config.h"
27987da915Sopenharmony_ci
28987da915Sopenharmony_ci#ifdef HAVE_STDIO_H
29987da915Sopenharmony_ci#include <stdio.h>
30987da915Sopenharmony_ci#endif
31987da915Sopenharmony_ci#ifdef HAVE_GETOPT_H
32987da915Sopenharmony_ci#include <getopt.h>
33987da915Sopenharmony_ci#endif
34987da915Sopenharmony_ci#ifdef HAVE_STDLIB_H
35987da915Sopenharmony_ci#include <stdlib.h>
36987da915Sopenharmony_ci#endif
37987da915Sopenharmony_ci#ifdef HAVE_STRING_H
38987da915Sopenharmony_ci#include <string.h>
39987da915Sopenharmony_ci#endif
40987da915Sopenharmony_ci#ifdef HAVE_LIMITS_H
41987da915Sopenharmony_ci#include <limits.h>
42987da915Sopenharmony_ci#endif
43987da915Sopenharmony_ci
44987da915Sopenharmony_ci#include "ntfscluster.h"
45987da915Sopenharmony_ci#include "types.h"
46987da915Sopenharmony_ci#include "attrib.h"
47987da915Sopenharmony_ci#include "utils.h"
48987da915Sopenharmony_ci#include "volume.h"
49987da915Sopenharmony_ci#include "debug.h"
50987da915Sopenharmony_ci#include "dir.h"
51987da915Sopenharmony_ci#include "cluster.h"
52987da915Sopenharmony_ci/* #include "version.h" */
53987da915Sopenharmony_ci#include "logging.h"
54987da915Sopenharmony_ci
55987da915Sopenharmony_cistatic const char *EXEC_NAME = "ntfscluster";
56987da915Sopenharmony_cistatic struct options opts;
57987da915Sopenharmony_ci
58987da915Sopenharmony_ci/**
59987da915Sopenharmony_ci * version - Print version information about the program
60987da915Sopenharmony_ci *
61987da915Sopenharmony_ci * Print a copyright statement and a brief description of the program.
62987da915Sopenharmony_ci *
63987da915Sopenharmony_ci * Return:  none
64987da915Sopenharmony_ci */
65987da915Sopenharmony_cistatic void version(void)
66987da915Sopenharmony_ci{
67987da915Sopenharmony_ci	ntfs_log_info("\n%s v%s (libntfs-3g) - Find the owner of any given sector or "
68987da915Sopenharmony_ci			"cluster.\n\n", EXEC_NAME, VERSION);
69987da915Sopenharmony_ci	ntfs_log_info("Copyright (c) 2002-2003 Richard Russon\n");
70987da915Sopenharmony_ci	ntfs_log_info("Copyright (c) 2005 Anton Altaparmakov\n");
71987da915Sopenharmony_ci	ntfs_log_info("Copyright (c) 2005-2006 Szabolcs Szakacsits\n");
72987da915Sopenharmony_ci	ntfs_log_info("\n%s\n%s%s\n", ntfs_gpl, ntfs_bugs, ntfs_home);
73987da915Sopenharmony_ci}
74987da915Sopenharmony_ci
75987da915Sopenharmony_ci/**
76987da915Sopenharmony_ci * usage - Print a list of the parameters to the program
77987da915Sopenharmony_ci *
78987da915Sopenharmony_ci * Print a list of the parameters and options for the program.
79987da915Sopenharmony_ci *
80987da915Sopenharmony_ci * Return:  none
81987da915Sopenharmony_ci */
82987da915Sopenharmony_cistatic void usage(void)
83987da915Sopenharmony_ci{
84987da915Sopenharmony_ci	ntfs_log_info("\nUsage: %s [options] device\n"
85987da915Sopenharmony_ci		"    -i, --info           Print information about the volume (default)\n"
86987da915Sopenharmony_ci		"\n"
87987da915Sopenharmony_ci		"    -c, --cluster RANGE  Look for objects in this range of clusters\n"
88987da915Sopenharmony_ci		"    -s, --sector RANGE   Look for objects in this range of sectors\n"
89987da915Sopenharmony_ci		"    -I, --inode NUM      Show information about this inode\n"
90987da915Sopenharmony_ci		"    -F, --filename NAME  Show information about this file\n"
91987da915Sopenharmony_ci	/*	"    -l, --last           Find the last file on the volume\n" */
92987da915Sopenharmony_ci		"\n"
93987da915Sopenharmony_ci		"    -f, --force          Use less caution\n"
94987da915Sopenharmony_ci		"    -q, --quiet          Less output\n"
95987da915Sopenharmony_ci		"    -v, --verbose        More output\n"
96987da915Sopenharmony_ci		"    -V, --version        Version information\n"
97987da915Sopenharmony_ci		"    -h, --help           Print this help\n\n",
98987da915Sopenharmony_ci		EXEC_NAME);
99987da915Sopenharmony_ci	ntfs_log_info("%s%s\n", ntfs_bugs, ntfs_home);
100987da915Sopenharmony_ci}
101987da915Sopenharmony_ci
102987da915Sopenharmony_ci/**
103987da915Sopenharmony_ci * parse_options - Read and validate the programs command line
104987da915Sopenharmony_ci *
105987da915Sopenharmony_ci * Read the command line, verify the syntax and parse the options.
106987da915Sopenharmony_ci * This function is very long, but quite simple.
107987da915Sopenharmony_ci *
108987da915Sopenharmony_ci * Return:  1 Success
109987da915Sopenharmony_ci *	    0 Error, one or more problems
110987da915Sopenharmony_ci */
111987da915Sopenharmony_cistatic int parse_options(int argc, char **argv)
112987da915Sopenharmony_ci{
113987da915Sopenharmony_ci	static const char *sopt = "-c:F:fh?I:ilqs:vV";
114987da915Sopenharmony_ci	static const struct option lopt[] = {
115987da915Sopenharmony_ci		{ "cluster",	required_argument,	NULL, 'c' },
116987da915Sopenharmony_ci		{ "filename",	required_argument,	NULL, 'F' },
117987da915Sopenharmony_ci		{ "force",	no_argument,		NULL, 'f' },
118987da915Sopenharmony_ci		{ "help",	no_argument,		NULL, 'h' },
119987da915Sopenharmony_ci		{ "info",	no_argument,		NULL, 'i' },
120987da915Sopenharmony_ci		{ "inode",	required_argument,	NULL, 'I' },
121987da915Sopenharmony_ci		{ "last",	no_argument,		NULL, 'l' },
122987da915Sopenharmony_ci		{ "quiet",	no_argument,		NULL, 'q' },
123987da915Sopenharmony_ci		{ "sector",	required_argument,	NULL, 's' },
124987da915Sopenharmony_ci		{ "verbose",	no_argument,		NULL, 'v' },
125987da915Sopenharmony_ci		{ "version",	no_argument,		NULL, 'V' },
126987da915Sopenharmony_ci		{ NULL,		0,			NULL, 0   }
127987da915Sopenharmony_ci	};
128987da915Sopenharmony_ci
129987da915Sopenharmony_ci	int c = -1;
130987da915Sopenharmony_ci	int err  = 0;
131987da915Sopenharmony_ci	int ver  = 0;
132987da915Sopenharmony_ci	int help = 0;
133987da915Sopenharmony_ci	int levels = 0;
134987da915Sopenharmony_ci	char *end = NULL;
135987da915Sopenharmony_ci
136987da915Sopenharmony_ci	opterr = 0; /* We'll handle the errors, thank you. */
137987da915Sopenharmony_ci
138987da915Sopenharmony_ci	opts.action      = act_none;
139987da915Sopenharmony_ci	opts.range_begin = -1;
140987da915Sopenharmony_ci	opts.range_end   = -1;
141987da915Sopenharmony_ci
142987da915Sopenharmony_ci	while ((c = getopt_long(argc, argv, sopt, lopt, NULL)) != -1) {
143987da915Sopenharmony_ci		switch (c) {
144987da915Sopenharmony_ci		case 1:	/* A non-option argument */
145987da915Sopenharmony_ci			if (!opts.device) {
146987da915Sopenharmony_ci				opts.device = argv[optind-1];
147987da915Sopenharmony_ci			} else {
148987da915Sopenharmony_ci				opts.device = NULL;
149987da915Sopenharmony_ci				err++;
150987da915Sopenharmony_ci			}
151987da915Sopenharmony_ci			break;
152987da915Sopenharmony_ci
153987da915Sopenharmony_ci		case 'c':
154987da915Sopenharmony_ci			if ((opts.action == act_none) &&
155987da915Sopenharmony_ci			    (utils_parse_range(optarg, &opts.range_begin, &opts.range_end, FALSE)))
156987da915Sopenharmony_ci				opts.action = act_cluster;
157987da915Sopenharmony_ci			else
158987da915Sopenharmony_ci				opts.action = act_error;
159987da915Sopenharmony_ci			break;
160987da915Sopenharmony_ci		case 'F':
161987da915Sopenharmony_ci			if (opts.action == act_none) {
162987da915Sopenharmony_ci				opts.action = act_file;
163987da915Sopenharmony_ci				opts.filename = optarg;
164987da915Sopenharmony_ci			} else {
165987da915Sopenharmony_ci				opts.action = act_error;
166987da915Sopenharmony_ci			}
167987da915Sopenharmony_ci			break;
168987da915Sopenharmony_ci		case 'f':
169987da915Sopenharmony_ci			opts.force++;
170987da915Sopenharmony_ci			break;
171987da915Sopenharmony_ci		case 'h':
172987da915Sopenharmony_ci			help++;
173987da915Sopenharmony_ci			break;
174987da915Sopenharmony_ci		case 'I':
175987da915Sopenharmony_ci			if (opts.action == act_none) {
176987da915Sopenharmony_ci				opts.action = act_inode;
177987da915Sopenharmony_ci				opts.inode = strtol(optarg, &end, 0);
178987da915Sopenharmony_ci				if (end && *end)
179987da915Sopenharmony_ci					err++;
180987da915Sopenharmony_ci			} else {
181987da915Sopenharmony_ci				opts.action = act_error;
182987da915Sopenharmony_ci			}
183987da915Sopenharmony_ci			break;
184987da915Sopenharmony_ci		case 'i':
185987da915Sopenharmony_ci			if (opts.action == act_none)
186987da915Sopenharmony_ci				opts.action = act_info;
187987da915Sopenharmony_ci			else
188987da915Sopenharmony_ci				opts.action = act_error;
189987da915Sopenharmony_ci			break;
190987da915Sopenharmony_ci		case 'l':
191987da915Sopenharmony_ci			if (opts.action == act_none)
192987da915Sopenharmony_ci				opts.action = act_last;
193987da915Sopenharmony_ci			else
194987da915Sopenharmony_ci				opts.action = act_error;
195987da915Sopenharmony_ci			break;
196987da915Sopenharmony_ci		case 'q':
197987da915Sopenharmony_ci			opts.quiet++;
198987da915Sopenharmony_ci			ntfs_log_clear_levels(NTFS_LOG_LEVEL_QUIET);
199987da915Sopenharmony_ci			break;
200987da915Sopenharmony_ci		case 's':
201987da915Sopenharmony_ci			if ((opts.action == act_none) &&
202987da915Sopenharmony_ci			    (utils_parse_range(optarg, &opts.range_begin, &opts.range_end, FALSE)))
203987da915Sopenharmony_ci				opts.action = act_sector;
204987da915Sopenharmony_ci			else
205987da915Sopenharmony_ci				opts.action = act_error;
206987da915Sopenharmony_ci			break;
207987da915Sopenharmony_ci		case 'v':
208987da915Sopenharmony_ci			opts.verbose++;
209987da915Sopenharmony_ci			ntfs_log_set_levels(NTFS_LOG_LEVEL_VERBOSE);
210987da915Sopenharmony_ci			break;
211987da915Sopenharmony_ci		case 'V':
212987da915Sopenharmony_ci			ver++;
213987da915Sopenharmony_ci			break;
214987da915Sopenharmony_ci		case '?':
215987da915Sopenharmony_ci			if (strncmp (argv[optind-1], "--log-", 6) == 0) {
216987da915Sopenharmony_ci				if (!ntfs_log_parse_option (argv[optind-1]))
217987da915Sopenharmony_ci					err++;
218987da915Sopenharmony_ci				break;
219987da915Sopenharmony_ci			}
220987da915Sopenharmony_ci			/* fall through */
221987da915Sopenharmony_ci		default:
222987da915Sopenharmony_ci			if ((optopt == 'c') || (optopt == 's'))
223987da915Sopenharmony_ci				ntfs_log_error("Option '%s' requires an argument.\n", argv[optind-1]);
224987da915Sopenharmony_ci			else
225987da915Sopenharmony_ci				ntfs_log_error("Unknown option '%s'.\n", argv[optind-1]);
226987da915Sopenharmony_ci			err++;
227987da915Sopenharmony_ci			break;
228987da915Sopenharmony_ci		}
229987da915Sopenharmony_ci	}
230987da915Sopenharmony_ci
231987da915Sopenharmony_ci	/* Make sure we're in sync with the log levels */
232987da915Sopenharmony_ci	levels = ntfs_log_get_levels();
233987da915Sopenharmony_ci	if (levels & NTFS_LOG_LEVEL_VERBOSE)
234987da915Sopenharmony_ci		opts.verbose++;
235987da915Sopenharmony_ci	if (!(levels & NTFS_LOG_LEVEL_QUIET))
236987da915Sopenharmony_ci		opts.quiet++;
237987da915Sopenharmony_ci
238987da915Sopenharmony_ci	if (help || ver) {
239987da915Sopenharmony_ci		opts.quiet = 0;
240987da915Sopenharmony_ci	} else {
241987da915Sopenharmony_ci		if (opts.action == act_none)
242987da915Sopenharmony_ci			opts.action = act_info;
243987da915Sopenharmony_ci		if (opts.action == act_info)
244987da915Sopenharmony_ci			opts.quiet = 0;
245987da915Sopenharmony_ci
246987da915Sopenharmony_ci		if (opts.device == NULL) {
247987da915Sopenharmony_ci			if (argc > 1)
248987da915Sopenharmony_ci				ntfs_log_error("You must specify exactly one device.\n");
249987da915Sopenharmony_ci			err++;
250987da915Sopenharmony_ci		}
251987da915Sopenharmony_ci
252987da915Sopenharmony_ci		if (opts.quiet && opts.verbose) {
253987da915Sopenharmony_ci			ntfs_log_error("You may not use --quiet and --verbose at the same time.\n");
254987da915Sopenharmony_ci			err++;
255987da915Sopenharmony_ci		}
256987da915Sopenharmony_ci
257987da915Sopenharmony_ci		if (opts.action == act_error) {
258987da915Sopenharmony_ci			ntfs_log_error("You may only specify one action: --info, --cluster, --sector or --last.\n");
259987da915Sopenharmony_ci			err++;
260987da915Sopenharmony_ci		} else if (opts.range_begin > opts.range_end) {
261987da915Sopenharmony_ci			ntfs_log_error("The range must be in ascending order.\n");
262987da915Sopenharmony_ci			err++;
263987da915Sopenharmony_ci		}
264987da915Sopenharmony_ci	}
265987da915Sopenharmony_ci
266987da915Sopenharmony_ci	if (ver)
267987da915Sopenharmony_ci		version();
268987da915Sopenharmony_ci	if (help || err)
269987da915Sopenharmony_ci		usage();
270987da915Sopenharmony_ci
271987da915Sopenharmony_ci		/* tri-state 0 : done, 1 : error, -1 : proceed */
272987da915Sopenharmony_ci	return (err ? 1 : (help || ver ? 0 : -1));
273987da915Sopenharmony_ci}
274987da915Sopenharmony_ci
275987da915Sopenharmony_ci
276987da915Sopenharmony_ci/**
277987da915Sopenharmony_ci * info
278987da915Sopenharmony_ci */
279987da915Sopenharmony_cistatic int info(ntfs_volume *vol)
280987da915Sopenharmony_ci{
281987da915Sopenharmony_ci	u64 a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p, q, r, s, t, u;
282987da915Sopenharmony_ci	int cb, sb, cps;
283987da915Sopenharmony_ci	u64 uc = 0, mc = 0, fc = 0;
284987da915Sopenharmony_ci
285987da915Sopenharmony_ci	struct mft_search_ctx *m_ctx;
286987da915Sopenharmony_ci	ntfs_attr_search_ctx *a_ctx;
287987da915Sopenharmony_ci	runlist_element *rl;
288987da915Sopenharmony_ci	ATTR_RECORD *rec;
289987da915Sopenharmony_ci	int z;
290987da915Sopenharmony_ci	int inuse = 0;
291987da915Sopenharmony_ci
292987da915Sopenharmony_ci	m_ctx = mft_get_search_ctx(vol);
293987da915Sopenharmony_ci	m_ctx->flags_search = FEMR_IN_USE | FEMR_METADATA | FEMR_BASE_RECORD | FEMR_NOT_BASE_RECORD;
294987da915Sopenharmony_ci	while (mft_next_record(m_ctx) == 0) {
295987da915Sopenharmony_ci
296987da915Sopenharmony_ci		if (!(m_ctx->flags_match & FEMR_IN_USE))
297987da915Sopenharmony_ci			continue;
298987da915Sopenharmony_ci
299987da915Sopenharmony_ci		inuse++;
300987da915Sopenharmony_ci
301987da915Sopenharmony_ci		a_ctx = ntfs_attr_get_search_ctx(m_ctx->inode, NULL);
302987da915Sopenharmony_ci
303987da915Sopenharmony_ci		while ((rec = find_attribute(AT_UNUSED, a_ctx))) {
304987da915Sopenharmony_ci
305987da915Sopenharmony_ci			if (!rec->non_resident)
306987da915Sopenharmony_ci				continue;
307987da915Sopenharmony_ci
308987da915Sopenharmony_ci			rl = ntfs_mapping_pairs_decompress(vol, rec, NULL);
309987da915Sopenharmony_ci
310987da915Sopenharmony_ci			for (z = 0; rl[z].length > 0; z++)
311987da915Sopenharmony_ci			{
312987da915Sopenharmony_ci				if (rl[z].lcn >= 0) {
313987da915Sopenharmony_ci					if (m_ctx->flags_match & FEMR_METADATA)
314987da915Sopenharmony_ci						mc += rl[z].length;
315987da915Sopenharmony_ci					else
316987da915Sopenharmony_ci						uc += rl[z].length;
317987da915Sopenharmony_ci				}
318987da915Sopenharmony_ci
319987da915Sopenharmony_ci			}
320987da915Sopenharmony_ci
321987da915Sopenharmony_ci			free(rl);
322987da915Sopenharmony_ci		}
323987da915Sopenharmony_ci
324987da915Sopenharmony_ci		ntfs_attr_put_search_ctx(a_ctx);
325987da915Sopenharmony_ci	}
326987da915Sopenharmony_ci	mft_put_search_ctx(m_ctx);
327987da915Sopenharmony_ci
328987da915Sopenharmony_ci	cb  = vol->cluster_size_bits;
329987da915Sopenharmony_ci	sb  = vol->sector_size_bits;
330987da915Sopenharmony_ci	cps = cb - sb;
331987da915Sopenharmony_ci
332987da915Sopenharmony_ci	fc  = vol->nr_clusters-mc-uc;
333987da915Sopenharmony_ci	fc  <<= cb;
334987da915Sopenharmony_ci	mc  <<= cb;
335987da915Sopenharmony_ci	uc  <<= cb;
336987da915Sopenharmony_ci
337987da915Sopenharmony_ci	a = vol->sector_size;
338987da915Sopenharmony_ci	b = vol->cluster_size;
339987da915Sopenharmony_ci	c = 1 << cps;
340987da915Sopenharmony_ci	d = vol->nr_clusters << cb;
341987da915Sopenharmony_ci	e = vol->nr_clusters;
342987da915Sopenharmony_ci	f = vol->nr_clusters >> cps;
343987da915Sopenharmony_ci	g = vol->mft_na->initialized_size >> vol->mft_record_size_bits;
344987da915Sopenharmony_ci	h = inuse;
345987da915Sopenharmony_ci	i = h * 100 / g;
346987da915Sopenharmony_ci	j = fc;
347987da915Sopenharmony_ci	k = fc >> sb;
348987da915Sopenharmony_ci	l = fc >> cb;
349987da915Sopenharmony_ci	m = fc * 100 / b / e;
350987da915Sopenharmony_ci	n = uc;
351987da915Sopenharmony_ci	o = uc >> sb;
352987da915Sopenharmony_ci	p = uc >> cb;
353987da915Sopenharmony_ci	q = uc * 100 / b / e;
354987da915Sopenharmony_ci	r = mc;
355987da915Sopenharmony_ci	s = mc >> sb;
356987da915Sopenharmony_ci	t = mc >> cb;
357987da915Sopenharmony_ci	u = mc * 100 / b / e;
358987da915Sopenharmony_ci
359987da915Sopenharmony_ci	ntfs_log_info("bytes per sector        : %llu\n", (unsigned long long)a);
360987da915Sopenharmony_ci	ntfs_log_info("bytes per cluster       : %llu\n", (unsigned long long)b);
361987da915Sopenharmony_ci	ntfs_log_info("sectors per cluster     : %llu\n", (unsigned long long)c);
362987da915Sopenharmony_ci	ntfs_log_info("bytes per volume        : %llu\n", (unsigned long long)d);
363987da915Sopenharmony_ci	ntfs_log_info("sectors per volume      : %llu\n", (unsigned long long)e);
364987da915Sopenharmony_ci	ntfs_log_info("clusters per volume     : %llu\n", (unsigned long long)f);
365987da915Sopenharmony_ci	ntfs_log_info("initialized mft records : %llu\n", (unsigned long long)g);
366987da915Sopenharmony_ci	ntfs_log_info("mft records in use      : %llu\n", (unsigned long long)h);
367987da915Sopenharmony_ci	ntfs_log_info("mft records percentage  : %llu\n", (unsigned long long)i);
368987da915Sopenharmony_ci	ntfs_log_info("bytes of free space     : %llu\n", (unsigned long long)j);
369987da915Sopenharmony_ci	ntfs_log_info("sectors of free space   : %llu\n", (unsigned long long)k);
370987da915Sopenharmony_ci	ntfs_log_info("clusters of free space  : %llu\n", (unsigned long long)l);
371987da915Sopenharmony_ci	ntfs_log_info("percentage free space   : %llu\n", (unsigned long long)m);
372987da915Sopenharmony_ci	ntfs_log_info("bytes of user data      : %llu\n", (unsigned long long)n);
373987da915Sopenharmony_ci	ntfs_log_info("sectors of user data    : %llu\n", (unsigned long long)o);
374987da915Sopenharmony_ci	ntfs_log_info("clusters of user data   : %llu\n", (unsigned long long)p);
375987da915Sopenharmony_ci	ntfs_log_info("percentage user data    : %llu\n", (unsigned long long)q);
376987da915Sopenharmony_ci	ntfs_log_info("bytes of metadata       : %llu\n", (unsigned long long)r);
377987da915Sopenharmony_ci	ntfs_log_info("sectors of metadata     : %llu\n", (unsigned long long)s);
378987da915Sopenharmony_ci	ntfs_log_info("clusters of metadata    : %llu\n", (unsigned long long)t);
379987da915Sopenharmony_ci	ntfs_log_info("percentage metadata     : %llu\n", (unsigned long long)u);
380987da915Sopenharmony_ci
381987da915Sopenharmony_ci	return 0;
382987da915Sopenharmony_ci}
383987da915Sopenharmony_ci
384987da915Sopenharmony_ci/**
385987da915Sopenharmony_ci * dump_file
386987da915Sopenharmony_ci */
387987da915Sopenharmony_cistatic int dump_file(ntfs_volume *vol, ntfs_inode *ino)
388987da915Sopenharmony_ci{
389987da915Sopenharmony_ci	char buffer[1024];
390987da915Sopenharmony_ci	ntfs_attr_search_ctx *ctx;
391987da915Sopenharmony_ci	ATTR_RECORD *rec;
392987da915Sopenharmony_ci	int i;
393987da915Sopenharmony_ci	runlist *runs;
394987da915Sopenharmony_ci
395987da915Sopenharmony_ci	utils_inode_get_name(ino, buffer, sizeof(buffer));
396987da915Sopenharmony_ci
397987da915Sopenharmony_ci	ntfs_log_info("Dump: %s\n", buffer);
398987da915Sopenharmony_ci
399987da915Sopenharmony_ci	ctx = ntfs_attr_get_search_ctx(ino, NULL);
400987da915Sopenharmony_ci
401987da915Sopenharmony_ci	while ((rec = find_attribute(AT_UNUSED, ctx))) {
402987da915Sopenharmony_ci		ntfs_log_info("    0x%02x - ", (int)le32_to_cpu(rec->type));
403987da915Sopenharmony_ci		if (rec->non_resident) {
404987da915Sopenharmony_ci			ntfs_log_info("non-resident\n");
405987da915Sopenharmony_ci			runs = ntfs_mapping_pairs_decompress(vol, rec, NULL);
406987da915Sopenharmony_ci			if (runs) {
407987da915Sopenharmony_ci				ntfs_log_info("             VCN     LCN     Length\n");
408987da915Sopenharmony_ci				for (i = 0; runs[i].length > 0; i++) {
409987da915Sopenharmony_ci					ntfs_log_info("        %8lld %8lld %8lld\n",
410987da915Sopenharmony_ci							(long long)runs[i].vcn,
411987da915Sopenharmony_ci							(long long)runs[i].lcn,
412987da915Sopenharmony_ci							(long long)
413987da915Sopenharmony_ci							runs[i].length);
414987da915Sopenharmony_ci				}
415987da915Sopenharmony_ci				free(runs);
416987da915Sopenharmony_ci			}
417987da915Sopenharmony_ci		} else {
418987da915Sopenharmony_ci			ntfs_log_info("resident\n");
419987da915Sopenharmony_ci		}
420987da915Sopenharmony_ci	}
421987da915Sopenharmony_ci
422987da915Sopenharmony_ci	ntfs_attr_put_search_ctx(ctx);
423987da915Sopenharmony_ci	return 0;
424987da915Sopenharmony_ci}
425987da915Sopenharmony_ci
426987da915Sopenharmony_ci/**
427987da915Sopenharmony_ci * print_match
428987da915Sopenharmony_ci */
429987da915Sopenharmony_cistatic int print_match(ntfs_inode *ino, ATTR_RECORD *attr,
430987da915Sopenharmony_ci	runlist_element *run, void *data __attribute__((unused)))
431987da915Sopenharmony_ci{
432987da915Sopenharmony_ci	char *buffer;
433987da915Sopenharmony_ci
434987da915Sopenharmony_ci	if (!ino || !attr || !run)
435987da915Sopenharmony_ci		return 1;
436987da915Sopenharmony_ci
437987da915Sopenharmony_ci	buffer = malloc(MAX_PATH);
438987da915Sopenharmony_ci	if (!buffer) {
439987da915Sopenharmony_ci		ntfs_log_error("!buffer\n");
440987da915Sopenharmony_ci		return 1;
441987da915Sopenharmony_ci	}
442987da915Sopenharmony_ci
443987da915Sopenharmony_ci	utils_inode_get_name(ino, buffer, MAX_PATH);
444987da915Sopenharmony_ci	ntfs_log_info("Inode %llu %s", (unsigned long long)ino->mft_no, buffer);
445987da915Sopenharmony_ci
446987da915Sopenharmony_ci	utils_attr_get_name(ino->vol, attr, buffer, MAX_PATH);
447987da915Sopenharmony_ci	ntfs_log_info("/%s\n", buffer);
448987da915Sopenharmony_ci
449987da915Sopenharmony_ci	free(buffer);
450987da915Sopenharmony_ci	return 0;
451987da915Sopenharmony_ci}
452987da915Sopenharmony_ci
453987da915Sopenharmony_ci/**
454987da915Sopenharmony_ci * find_last
455987da915Sopenharmony_ci */
456987da915Sopenharmony_cistatic int find_last(ntfs_inode *ino, ATTR_RECORD *attr, runlist_element *run,
457987da915Sopenharmony_ci	void *data)
458987da915Sopenharmony_ci{
459987da915Sopenharmony_ci	struct match *m;
460987da915Sopenharmony_ci
461987da915Sopenharmony_ci	if (!ino || !attr || !run || !data)
462987da915Sopenharmony_ci		return 1;
463987da915Sopenharmony_ci
464987da915Sopenharmony_ci	m = data;
465987da915Sopenharmony_ci
466987da915Sopenharmony_ci	if ((run->lcn + run->length) > m->lcn) {
467987da915Sopenharmony_ci		m->inum = ino->mft_no;
468987da915Sopenharmony_ci		m->lcn  = run->lcn + run->length;
469987da915Sopenharmony_ci	}
470987da915Sopenharmony_ci
471987da915Sopenharmony_ci	return 0;
472987da915Sopenharmony_ci}
473987da915Sopenharmony_ci
474987da915Sopenharmony_ci/**
475987da915Sopenharmony_ci * main - Begin here
476987da915Sopenharmony_ci *
477987da915Sopenharmony_ci * Start from here.
478987da915Sopenharmony_ci *
479987da915Sopenharmony_ci * Return:  0  Success, the program worked
480987da915Sopenharmony_ci *	    1  Error, something went wrong
481987da915Sopenharmony_ci */
482987da915Sopenharmony_ciint main(int argc, char *argv[])
483987da915Sopenharmony_ci{
484987da915Sopenharmony_ci	ntfs_volume *vol;
485987da915Sopenharmony_ci	ntfs_inode *ino = NULL;
486987da915Sopenharmony_ci	struct match m;
487987da915Sopenharmony_ci	int res;
488987da915Sopenharmony_ci	int result = 1;
489987da915Sopenharmony_ci#ifdef HAVE_WINDOWS_H
490987da915Sopenharmony_ci	char *unix_name;
491987da915Sopenharmony_ci#endif
492987da915Sopenharmony_ci
493987da915Sopenharmony_ci	ntfs_log_set_handler(ntfs_log_handler_outerr);
494987da915Sopenharmony_ci
495987da915Sopenharmony_ci	res = parse_options(argc, argv);
496987da915Sopenharmony_ci	if (res >= 0)
497987da915Sopenharmony_ci		return (res);
498987da915Sopenharmony_ci
499987da915Sopenharmony_ci	utils_set_locale();
500987da915Sopenharmony_ci
501987da915Sopenharmony_ci	vol = utils_mount_volume(opts.device, NTFS_MNT_RDONLY |
502987da915Sopenharmony_ci			(opts.force ? NTFS_MNT_RECOVER : 0));
503987da915Sopenharmony_ci	if (!vol)
504987da915Sopenharmony_ci		return 1;
505987da915Sopenharmony_ci
506987da915Sopenharmony_ci	switch (opts.action) {
507987da915Sopenharmony_ci		case act_sector:
508987da915Sopenharmony_ci			if (opts.range_begin == opts.range_end)
509987da915Sopenharmony_ci				ntfs_log_quiet("Searching for sector %llu\n",
510987da915Sopenharmony_ci						(unsigned long long)opts.range_begin);
511987da915Sopenharmony_ci			else
512987da915Sopenharmony_ci				ntfs_log_quiet("Searching for sector range %llu-%llu\n", (unsigned long long)opts.range_begin, (unsigned long long)opts.range_end);
513987da915Sopenharmony_ci			/* Convert to clusters */
514987da915Sopenharmony_ci			opts.range_begin >>= (vol->cluster_size_bits - vol->sector_size_bits);
515987da915Sopenharmony_ci			opts.range_end   >>= (vol->cluster_size_bits - vol->sector_size_bits);
516987da915Sopenharmony_ci			result = cluster_find(vol, opts.range_begin, opts.range_end, (cluster_cb*)&print_match, NULL);
517987da915Sopenharmony_ci			break;
518987da915Sopenharmony_ci		case act_cluster:
519987da915Sopenharmony_ci			if (opts.range_begin == opts.range_end)
520987da915Sopenharmony_ci				ntfs_log_quiet("Searching for cluster %llu\n",
521987da915Sopenharmony_ci						(unsigned long long)opts.range_begin);
522987da915Sopenharmony_ci			else
523987da915Sopenharmony_ci				ntfs_log_quiet("Searching for cluster range %llu-%llu\n", (unsigned long long)opts.range_begin, (unsigned long long)opts.range_end);
524987da915Sopenharmony_ci			result = cluster_find(vol, opts.range_begin, opts.range_end, (cluster_cb*)&print_match, NULL);
525987da915Sopenharmony_ci			break;
526987da915Sopenharmony_ci		case act_file:
527987da915Sopenharmony_ci#ifdef HAVE_WINDOWS_H
528987da915Sopenharmony_ci			unix_name = ntfs_utils_unix_path(opts.filename);
529987da915Sopenharmony_ci			ino = 0;
530987da915Sopenharmony_ci			if (unix_name) {
531987da915Sopenharmony_ci				ino = ntfs_pathname_to_inode(vol, NULL,
532987da915Sopenharmony_ci								unix_name);
533987da915Sopenharmony_ci				free(unix_name);
534987da915Sopenharmony_ci			}
535987da915Sopenharmony_ci#else
536987da915Sopenharmony_ci			ino = ntfs_pathname_to_inode(vol, NULL, opts.filename);
537987da915Sopenharmony_ci#endif
538987da915Sopenharmony_ci			if (ino)
539987da915Sopenharmony_ci				result = dump_file(vol, ino);
540987da915Sopenharmony_ci			break;
541987da915Sopenharmony_ci		case act_inode:
542987da915Sopenharmony_ci			ino = ntfs_inode_open(vol, opts.inode);
543987da915Sopenharmony_ci			if (ino) {
544987da915Sopenharmony_ci				result = dump_file(vol, ino);
545987da915Sopenharmony_ci				ntfs_inode_close(ino);
546987da915Sopenharmony_ci			} else {
547987da915Sopenharmony_ci				ntfs_log_error("Cannot open inode %llu\n",
548987da915Sopenharmony_ci						(unsigned long long)opts.inode);
549987da915Sopenharmony_ci			}
550987da915Sopenharmony_ci			break;
551987da915Sopenharmony_ci		case act_last:
552987da915Sopenharmony_ci			memset(&m, 0, sizeof(m));
553987da915Sopenharmony_ci			m.lcn = -1;
554987da915Sopenharmony_ci			result = cluster_find(vol, 0, LONG_MAX, (cluster_cb*)&find_last, &m);
555987da915Sopenharmony_ci			if (m.lcn >= 0) {
556987da915Sopenharmony_ci				ino = ntfs_inode_open(vol, m.inum);
557987da915Sopenharmony_ci				if (ino) {
558987da915Sopenharmony_ci					result = dump_file(vol, ino);
559987da915Sopenharmony_ci					ntfs_inode_close(ino);
560987da915Sopenharmony_ci				} else {
561987da915Sopenharmony_ci					ntfs_log_error("Cannot open inode %llu\n",
562987da915Sopenharmony_ci							(unsigned long long)
563987da915Sopenharmony_ci							opts.inode);
564987da915Sopenharmony_ci				}
565987da915Sopenharmony_ci				result = 0;
566987da915Sopenharmony_ci			} else {
567987da915Sopenharmony_ci				result = 1;
568987da915Sopenharmony_ci			}
569987da915Sopenharmony_ci			break;
570987da915Sopenharmony_ci		case act_info:
571987da915Sopenharmony_ci		default:
572987da915Sopenharmony_ci			result = info(vol);
573987da915Sopenharmony_ci			break;
574987da915Sopenharmony_ci	}
575987da915Sopenharmony_ci
576987da915Sopenharmony_ci	ntfs_umount(vol, FALSE);
577987da915Sopenharmony_ci	return result;
578987da915Sopenharmony_ci}
579987da915Sopenharmony_ci
580987da915Sopenharmony_ci
581