1987da915Sopenharmony_ci/**
2987da915Sopenharmony_ci * ntfscat - Part of the Linux-NTFS project.
3987da915Sopenharmony_ci *
4987da915Sopenharmony_ci * Copyright (c) 2003-2005 Richard Russon
5987da915Sopenharmony_ci * Copyright (c) 2003-2005 Anton Altaparmakov
6987da915Sopenharmony_ci * Copyright (c) 2003-2005 Szabolcs Szakacsits
7987da915Sopenharmony_ci * Copyright (c) 2007      Yura Pakhuchiy
8987da915Sopenharmony_ci *
9987da915Sopenharmony_ci * This utility will concatenate files and print on the standard output.
10987da915Sopenharmony_ci *
11987da915Sopenharmony_ci * This program is free software; you can redistribute it and/or modify
12987da915Sopenharmony_ci * it under the terms of the GNU General Public License as published by
13987da915Sopenharmony_ci * the Free Software Foundation; either version 2 of the License, or
14987da915Sopenharmony_ci * (at your option) any later version.
15987da915Sopenharmony_ci *
16987da915Sopenharmony_ci * This program is distributed in the hope that it will be useful,
17987da915Sopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
18987da915Sopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19987da915Sopenharmony_ci * GNU General Public License for more details.
20987da915Sopenharmony_ci *
21987da915Sopenharmony_ci * You should have received a copy of the GNU General Public License
22987da915Sopenharmony_ci * along with this program (in the main directory of the Linux-NTFS
23987da915Sopenharmony_ci * distribution in the file COPYING); if not, write to the Free Software
24987da915Sopenharmony_ci * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
25987da915Sopenharmony_ci */
26987da915Sopenharmony_ci
27987da915Sopenharmony_ci#include "config.h"
28987da915Sopenharmony_ci
29987da915Sopenharmony_ci#ifdef HAVE_STDIO_H
30987da915Sopenharmony_ci#include <stdio.h>
31987da915Sopenharmony_ci#endif
32987da915Sopenharmony_ci#ifdef HAVE_GETOPT_H
33987da915Sopenharmony_ci#include <getopt.h>
34987da915Sopenharmony_ci#endif
35987da915Sopenharmony_ci#ifdef HAVE_STDLIB_H
36987da915Sopenharmony_ci#include <stdlib.h>
37987da915Sopenharmony_ci#endif
38987da915Sopenharmony_ci#ifdef HAVE_STRING_H
39987da915Sopenharmony_ci#include <string.h>
40987da915Sopenharmony_ci#endif
41987da915Sopenharmony_ci
42987da915Sopenharmony_ci#include "types.h"
43987da915Sopenharmony_ci#include "attrib.h"
44987da915Sopenharmony_ci#include "utils.h"
45987da915Sopenharmony_ci#include "volume.h"
46987da915Sopenharmony_ci#include "debug.h"
47987da915Sopenharmony_ci#include "dir.h"
48987da915Sopenharmony_ci#include "ntfscat.h"
49987da915Sopenharmony_ci/* #include "version.h" */
50987da915Sopenharmony_ci#include "utils.h"
51987da915Sopenharmony_ci
52987da915Sopenharmony_cistatic const char *EXEC_NAME = "ntfscat";
53987da915Sopenharmony_cistatic struct options opts;
54987da915Sopenharmony_ci
55987da915Sopenharmony_ci/**
56987da915Sopenharmony_ci * version - Print version information about the program
57987da915Sopenharmony_ci *
58987da915Sopenharmony_ci * Print a copyright statement and a brief description of the program.
59987da915Sopenharmony_ci *
60987da915Sopenharmony_ci * Return:  none
61987da915Sopenharmony_ci */
62987da915Sopenharmony_cistatic void version(void)
63987da915Sopenharmony_ci{
64987da915Sopenharmony_ci	ntfs_log_info("\n%s v%s (libntfs-3g) - Concatenate files and print "
65987da915Sopenharmony_ci			"on the standard output.\n\n", EXEC_NAME, VERSION);
66987da915Sopenharmony_ci	ntfs_log_info("Copyright (c) 2003-2005 Richard Russon\n");
67987da915Sopenharmony_ci	ntfs_log_info("Copyright (c) 2003-2005 Anton Altaparmakov\n");
68987da915Sopenharmony_ci	ntfs_log_info("Copyright (c) 2003-2005 Szabolcs Szakacsits\n");
69987da915Sopenharmony_ci	ntfs_log_info("Copyright (c) 2007      Yura Pakhuchiy\n");
70987da915Sopenharmony_ci	ntfs_log_info("\n%s\n%s%s\n", ntfs_gpl, ntfs_bugs, ntfs_home);
71987da915Sopenharmony_ci}
72987da915Sopenharmony_ci
73987da915Sopenharmony_ci/**
74987da915Sopenharmony_ci * usage - Print a list of the parameters to the program
75987da915Sopenharmony_ci *
76987da915Sopenharmony_ci * Print a list of the parameters and options for the program.
77987da915Sopenharmony_ci *
78987da915Sopenharmony_ci * Return:  none
79987da915Sopenharmony_ci */
80987da915Sopenharmony_cistatic void usage(void)
81987da915Sopenharmony_ci{
82987da915Sopenharmony_ci	ntfs_log_info("\nUsage: %s [options] device [file]\n\n"
83987da915Sopenharmony_ci		"    -a, --attribute TYPE       Display this attribute type\n"
84987da915Sopenharmony_ci		"    -n, --attribute-name NAME  Display this attribute name\n"
85987da915Sopenharmony_ci		"    -i, --inode NUM            Display this inode\n\n"
86987da915Sopenharmony_ci		"    -f, --force                Use less caution\n"
87987da915Sopenharmony_ci		"    -h, --help                 Print this help\n"
88987da915Sopenharmony_ci		"    -q, --quiet                Less output\n"
89987da915Sopenharmony_ci		"    -V, --version              Version information\n"
90987da915Sopenharmony_ci		"    -v, --verbose              More output\n\n",
91987da915Sopenharmony_ci// Does not work for compressed files at present so leave undocumented...
92987da915Sopenharmony_ci//		"    -r  --raw                  Display the raw data (e.g. for compressed or encrypted file)",
93987da915Sopenharmony_ci		EXEC_NAME);
94987da915Sopenharmony_ci	ntfs_log_info("%s%s\n", ntfs_bugs, ntfs_home);
95987da915Sopenharmony_ci}
96987da915Sopenharmony_ci
97987da915Sopenharmony_ci/**
98987da915Sopenharmony_ci * parse_attribute - Read an attribute name, or number
99987da915Sopenharmony_ci * @value:   String to be parsed
100987da915Sopenharmony_ci * @attr:    Resulting attribute id (on success)
101987da915Sopenharmony_ci *
102987da915Sopenharmony_ci * Read a string representing an attribute.  It may be a decimal, octal or
103987da915Sopenharmony_ci * hexadecimal number, or the attribute name in full.  The leading $ sign is
104987da915Sopenharmony_ci * optional.
105987da915Sopenharmony_ci *
106987da915Sopenharmony_ci * Return:  1  Success, a valid attribute name or number
107987da915Sopenharmony_ci *	    0  Error, not an attribute name or number
108987da915Sopenharmony_ci */
109987da915Sopenharmony_cistatic int parse_attribute(const char *value, ATTR_TYPES *attr)
110987da915Sopenharmony_ci{
111987da915Sopenharmony_ci	static const char *attr_name[] = {
112987da915Sopenharmony_ci		"$STANDARD_INFORMATION",
113987da915Sopenharmony_ci		"$ATTRIBUTE_LIST",
114987da915Sopenharmony_ci		"$FILE_NAME",
115987da915Sopenharmony_ci		"$OBJECT_ID",
116987da915Sopenharmony_ci		"$SECURITY_DESCRIPTOR",
117987da915Sopenharmony_ci		"$VOLUME_NAME",
118987da915Sopenharmony_ci		"$VOLUME_INFORMATION",
119987da915Sopenharmony_ci		"$DATA",
120987da915Sopenharmony_ci		"$INDEX_ROOT",
121987da915Sopenharmony_ci		"$INDEX_ALLOCATION",
122987da915Sopenharmony_ci		"$BITMAP",
123987da915Sopenharmony_ci		"$REPARSE_POINT",
124987da915Sopenharmony_ci		"$EA_INFORMATION",
125987da915Sopenharmony_ci		"$EA",
126987da915Sopenharmony_ci		"$PROPERTY_SET",
127987da915Sopenharmony_ci		"$LOGGED_UTILITY_STREAM",
128987da915Sopenharmony_ci		NULL
129987da915Sopenharmony_ci	};
130987da915Sopenharmony_ci
131987da915Sopenharmony_ci	int i;
132987da915Sopenharmony_ci	long num;
133987da915Sopenharmony_ci
134987da915Sopenharmony_ci	for (i = 0; attr_name[i]; i++) {
135987da915Sopenharmony_ci		if ((strcmp(value, attr_name[i]) == 0) ||
136987da915Sopenharmony_ci		    (strcmp(value, attr_name[i] + 1) == 0)) {
137987da915Sopenharmony_ci			*attr = (ATTR_TYPES)cpu_to_le32((i + 1) * 16);
138987da915Sopenharmony_ci			return 1;
139987da915Sopenharmony_ci		}
140987da915Sopenharmony_ci	}
141987da915Sopenharmony_ci
142987da915Sopenharmony_ci	num = strtol(value, NULL, 0);
143987da915Sopenharmony_ci	if ((num > 0) && (num < 257)) {
144987da915Sopenharmony_ci		*attr = (ATTR_TYPES)cpu_to_le32(num);
145987da915Sopenharmony_ci		return 1;
146987da915Sopenharmony_ci	}
147987da915Sopenharmony_ci
148987da915Sopenharmony_ci	return 0;
149987da915Sopenharmony_ci}
150987da915Sopenharmony_ci
151987da915Sopenharmony_ci/**
152987da915Sopenharmony_ci * parse_options - Read and validate the programs command line
153987da915Sopenharmony_ci *
154987da915Sopenharmony_ci * Read the command line, verify the syntax and parse the options.
155987da915Sopenharmony_ci * This function is very long, but quite simple.
156987da915Sopenharmony_ci *
157987da915Sopenharmony_ci * Return:  1 Success
158987da915Sopenharmony_ci *	    0 Error, one or more problems
159987da915Sopenharmony_ci */
160987da915Sopenharmony_cistatic int parse_options(int argc, char **argv)
161987da915Sopenharmony_ci{
162987da915Sopenharmony_ci	static const char *sopt = "-a:fh?i:n:qVvr";
163987da915Sopenharmony_ci	static const struct option lopt[] = {
164987da915Sopenharmony_ci		{ "attribute",      required_argument,	NULL, 'a' },
165987da915Sopenharmony_ci		{ "attribute-name", required_argument,	NULL, 'n' },
166987da915Sopenharmony_ci		{ "force",	    no_argument,	NULL, 'f' },
167987da915Sopenharmony_ci		{ "help",	    no_argument,	NULL, 'h' },
168987da915Sopenharmony_ci		{ "inode",	    required_argument,	NULL, 'i' },
169987da915Sopenharmony_ci		{ "quiet",	    no_argument,	NULL, 'q' },
170987da915Sopenharmony_ci		{ "version",	    no_argument,	NULL, 'V' },
171987da915Sopenharmony_ci		{ "verbose",	    no_argument,	NULL, 'v' },
172987da915Sopenharmony_ci		{ "raw",	    no_argument,	NULL, 'r' },
173987da915Sopenharmony_ci		{ NULL,		    0,			NULL, 0   }
174987da915Sopenharmony_ci	};
175987da915Sopenharmony_ci
176987da915Sopenharmony_ci	int c = -1;
177987da915Sopenharmony_ci	int err  = 0;
178987da915Sopenharmony_ci	int ver  = 0;
179987da915Sopenharmony_ci	int help = 0;
180987da915Sopenharmony_ci	int levels = 0;
181987da915Sopenharmony_ci	ATTR_TYPES attr = AT_UNUSED;
182987da915Sopenharmony_ci
183987da915Sopenharmony_ci	opterr = 0; /* We'll handle the errors, thank you. */
184987da915Sopenharmony_ci
185987da915Sopenharmony_ci	opts.inode = -1;
186987da915Sopenharmony_ci	opts.attr = const_cpu_to_le32(-1);
187987da915Sopenharmony_ci	opts.attr_name = NULL;
188987da915Sopenharmony_ci	opts.attr_name_len = 0;
189987da915Sopenharmony_ci
190987da915Sopenharmony_ci	while ((c = getopt_long(argc, argv, sopt, lopt, NULL)) != -1) {
191987da915Sopenharmony_ci		switch (c) {
192987da915Sopenharmony_ci		case 1:	/* A non-option argument */
193987da915Sopenharmony_ci			if (!opts.device) {
194987da915Sopenharmony_ci				opts.device = argv[optind - 1];
195987da915Sopenharmony_ci			} else if (!opts.file) {
196987da915Sopenharmony_ci				opts.file = argv[optind - 1];
197987da915Sopenharmony_ci			} else {
198987da915Sopenharmony_ci				ntfs_log_error("You must specify exactly one "
199987da915Sopenharmony_ci						"file.\n");
200987da915Sopenharmony_ci				err++;
201987da915Sopenharmony_ci			}
202987da915Sopenharmony_ci			break;
203987da915Sopenharmony_ci		case 'a':
204987da915Sopenharmony_ci			if (opts.attr != const_cpu_to_le32(-1)) {
205987da915Sopenharmony_ci				ntfs_log_error("You must specify exactly one "
206987da915Sopenharmony_ci						"attribute.\n");
207987da915Sopenharmony_ci			} else if (parse_attribute(optarg, &attr) > 0) {
208987da915Sopenharmony_ci				opts.attr = attr;
209987da915Sopenharmony_ci				break;
210987da915Sopenharmony_ci			} else {
211987da915Sopenharmony_ci				ntfs_log_error("Couldn't parse attribute.\n");
212987da915Sopenharmony_ci			}
213987da915Sopenharmony_ci			err++;
214987da915Sopenharmony_ci			break;
215987da915Sopenharmony_ci		case 'f':
216987da915Sopenharmony_ci			opts.force++;
217987da915Sopenharmony_ci			break;
218987da915Sopenharmony_ci		case 'h':
219987da915Sopenharmony_ci			help++;
220987da915Sopenharmony_ci			break;
221987da915Sopenharmony_ci		case 'i':
222987da915Sopenharmony_ci			if (opts.inode != -1)
223987da915Sopenharmony_ci				ntfs_log_error("You must specify exactly one inode.\n");
224987da915Sopenharmony_ci			else if (utils_parse_size(optarg, &opts.inode, FALSE))
225987da915Sopenharmony_ci				break;
226987da915Sopenharmony_ci			else
227987da915Sopenharmony_ci				ntfs_log_error("Couldn't parse inode number.\n");
228987da915Sopenharmony_ci			err++;
229987da915Sopenharmony_ci			break;
230987da915Sopenharmony_ci
231987da915Sopenharmony_ci		case 'n':
232987da915Sopenharmony_ci			opts.attr_name_len = ntfs_mbstoucs(optarg,
233987da915Sopenharmony_ci							   &opts.attr_name);
234987da915Sopenharmony_ci			if (opts.attr_name_len < 0) {
235987da915Sopenharmony_ci				ntfs_log_perror("Invalid attribute name '%s'",
236987da915Sopenharmony_ci						optarg);
237987da915Sopenharmony_ci				usage();
238987da915Sopenharmony_ci			}
239987da915Sopenharmony_ci			break;
240987da915Sopenharmony_ci
241987da915Sopenharmony_ci		case 'q':
242987da915Sopenharmony_ci			opts.quiet++;
243987da915Sopenharmony_ci			ntfs_log_clear_levels(NTFS_LOG_LEVEL_QUIET);
244987da915Sopenharmony_ci			break;
245987da915Sopenharmony_ci		case 'V':
246987da915Sopenharmony_ci			ver++;
247987da915Sopenharmony_ci			break;
248987da915Sopenharmony_ci		case 'v':
249987da915Sopenharmony_ci			opts.verbose++;
250987da915Sopenharmony_ci			ntfs_log_set_levels(NTFS_LOG_LEVEL_VERBOSE);
251987da915Sopenharmony_ci			break;
252987da915Sopenharmony_ci		case 'r':
253987da915Sopenharmony_ci			opts.raw = TRUE;
254987da915Sopenharmony_ci			break;
255987da915Sopenharmony_ci		case '?':
256987da915Sopenharmony_ci			if (strncmp (argv[optind-1], "--log-", 6) == 0) {
257987da915Sopenharmony_ci				if (!ntfs_log_parse_option (argv[optind-1]))
258987da915Sopenharmony_ci					err++;
259987da915Sopenharmony_ci				break;
260987da915Sopenharmony_ci			}
261987da915Sopenharmony_ci			/* fall through */
262987da915Sopenharmony_ci		default:
263987da915Sopenharmony_ci			ntfs_log_error("Unknown option '%s'.\n", argv[optind-1]);
264987da915Sopenharmony_ci			err++;
265987da915Sopenharmony_ci			break;
266987da915Sopenharmony_ci		}
267987da915Sopenharmony_ci	}
268987da915Sopenharmony_ci
269987da915Sopenharmony_ci	/* Make sure we're in sync with the log levels */
270987da915Sopenharmony_ci	levels = ntfs_log_get_levels();
271987da915Sopenharmony_ci	if (levels & NTFS_LOG_LEVEL_VERBOSE)
272987da915Sopenharmony_ci		opts.verbose++;
273987da915Sopenharmony_ci	if (!(levels & NTFS_LOG_LEVEL_QUIET))
274987da915Sopenharmony_ci		opts.quiet++;
275987da915Sopenharmony_ci
276987da915Sopenharmony_ci	if (help || ver) {
277987da915Sopenharmony_ci		opts.quiet = 0;
278987da915Sopenharmony_ci	} else {
279987da915Sopenharmony_ci		if (opts.device == NULL) {
280987da915Sopenharmony_ci			ntfs_log_error("You must specify a device.\n");
281987da915Sopenharmony_ci			err++;
282987da915Sopenharmony_ci
283987da915Sopenharmony_ci		} else if (opts.file == NULL && opts.inode == -1) {
284987da915Sopenharmony_ci			ntfs_log_error("You must specify a file or inode "
285987da915Sopenharmony_ci				 "with the -i option.\n");
286987da915Sopenharmony_ci			err++;
287987da915Sopenharmony_ci
288987da915Sopenharmony_ci		} else if (opts.file != NULL && opts.inode != -1) {
289987da915Sopenharmony_ci			ntfs_log_error("You can't specify both a file and inode.\n");
290987da915Sopenharmony_ci			err++;
291987da915Sopenharmony_ci		}
292987da915Sopenharmony_ci
293987da915Sopenharmony_ci		if (opts.quiet && opts.verbose) {
294987da915Sopenharmony_ci			ntfs_log_error("You may not use --quiet and --verbose at the "
295987da915Sopenharmony_ci					"same time.\n");
296987da915Sopenharmony_ci			err++;
297987da915Sopenharmony_ci		}
298987da915Sopenharmony_ci	}
299987da915Sopenharmony_ci
300987da915Sopenharmony_ci	if (ver)
301987da915Sopenharmony_ci		version();
302987da915Sopenharmony_ci	if (help || err)
303987da915Sopenharmony_ci		usage();
304987da915Sopenharmony_ci
305987da915Sopenharmony_ci		/* tri-state 0 : done, 1 : error, -1 : proceed */
306987da915Sopenharmony_ci	return (err ? 1 : (help || ver ? 0 : -1));
307987da915Sopenharmony_ci}
308987da915Sopenharmony_ci
309987da915Sopenharmony_ci/**
310987da915Sopenharmony_ci * index_get_size - Find the INDX block size from the index root
311987da915Sopenharmony_ci * @inode:  Inode of the directory to be checked
312987da915Sopenharmony_ci *
313987da915Sopenharmony_ci * Find the size of a directory's INDX block from the INDEX_ROOT attribute.
314987da915Sopenharmony_ci *
315987da915Sopenharmony_ci * Return:  n  Success, the INDX blocks are n bytes in size
316987da915Sopenharmony_ci *	    0  Error, not a directory
317987da915Sopenharmony_ci */
318987da915Sopenharmony_cistatic int index_get_size(ntfs_inode *inode)
319987da915Sopenharmony_ci{
320987da915Sopenharmony_ci	ATTR_RECORD *attr90;
321987da915Sopenharmony_ci	INDEX_ROOT *iroot;
322987da915Sopenharmony_ci
323987da915Sopenharmony_ci	attr90 = find_first_attribute(AT_INDEX_ROOT, inode->mrec);
324987da915Sopenharmony_ci	if (!attr90)
325987da915Sopenharmony_ci		return 0;	// not a directory
326987da915Sopenharmony_ci
327987da915Sopenharmony_ci	iroot = (INDEX_ROOT*)((u8*)attr90 + le16_to_cpu(attr90->value_offset));
328987da915Sopenharmony_ci	return le32_to_cpu(iroot->index_block_size);
329987da915Sopenharmony_ci}
330987da915Sopenharmony_ci
331987da915Sopenharmony_ci/**
332987da915Sopenharmony_ci * cat
333987da915Sopenharmony_ci */
334987da915Sopenharmony_cistatic int cat(ntfs_volume *vol, ntfs_inode *inode, ATTR_TYPES type,
335987da915Sopenharmony_ci		ntfschar *name, int namelen)
336987da915Sopenharmony_ci{
337987da915Sopenharmony_ci	const int bufsize = 4096;
338987da915Sopenharmony_ci	char *buffer;
339987da915Sopenharmony_ci	ntfs_attr *attr;
340987da915Sopenharmony_ci	s64 bytes_read, written;
341987da915Sopenharmony_ci	s64 offset;
342987da915Sopenharmony_ci	u32 block_size;
343987da915Sopenharmony_ci
344987da915Sopenharmony_ci	buffer = malloc(bufsize);
345987da915Sopenharmony_ci	if (!buffer)
346987da915Sopenharmony_ci		return 1;
347987da915Sopenharmony_ci
348987da915Sopenharmony_ci	attr = ntfs_attr_open(inode, type, name, namelen);
349987da915Sopenharmony_ci	if (!attr) {
350987da915Sopenharmony_ci		ntfs_log_error("Cannot find attribute type 0x%x.\n",
351987da915Sopenharmony_ci				le32_to_cpu(type));
352987da915Sopenharmony_ci		free(buffer);
353987da915Sopenharmony_ci		return 1;
354987da915Sopenharmony_ci	}
355987da915Sopenharmony_ci
356987da915Sopenharmony_ci	if ((inode->mft_no < 2) && (attr->type == AT_DATA))
357987da915Sopenharmony_ci		block_size = vol->mft_record_size;
358987da915Sopenharmony_ci	else if (attr->type == AT_INDEX_ALLOCATION)
359987da915Sopenharmony_ci		block_size = index_get_size(inode);
360987da915Sopenharmony_ci	else
361987da915Sopenharmony_ci		block_size = 0;
362987da915Sopenharmony_ci
363987da915Sopenharmony_ci	offset = 0;
364987da915Sopenharmony_ci	for (;;) {
365987da915Sopenharmony_ci		if (!opts.raw && block_size > 0) {
366987da915Sopenharmony_ci			// These types have fixup
367987da915Sopenharmony_ci			bytes_read = ntfs_attr_mst_pread(attr, offset, 1, block_size, buffer);
368987da915Sopenharmony_ci			if (bytes_read > 0)
369987da915Sopenharmony_ci				bytes_read *= block_size;
370987da915Sopenharmony_ci		} else {
371987da915Sopenharmony_ci			bytes_read = ntfs_attr_pread(attr, offset, bufsize, buffer);
372987da915Sopenharmony_ci		}
373987da915Sopenharmony_ci		//ntfs_log_info("read %lld bytes\n", bytes_read);
374987da915Sopenharmony_ci		if (bytes_read == -1) {
375987da915Sopenharmony_ci			ntfs_log_perror("ERROR: Couldn't read file");
376987da915Sopenharmony_ci			break;
377987da915Sopenharmony_ci		}
378987da915Sopenharmony_ci		if (!bytes_read)
379987da915Sopenharmony_ci			break;
380987da915Sopenharmony_ci
381987da915Sopenharmony_ci		written = fwrite(buffer, 1, bytes_read, stdout);
382987da915Sopenharmony_ci		if (written != bytes_read) {
383987da915Sopenharmony_ci			ntfs_log_perror("ERROR: Couldn't output all data!");
384987da915Sopenharmony_ci			break;
385987da915Sopenharmony_ci		}
386987da915Sopenharmony_ci		offset += bytes_read;
387987da915Sopenharmony_ci	}
388987da915Sopenharmony_ci
389987da915Sopenharmony_ci	ntfs_attr_close(attr);
390987da915Sopenharmony_ci	free(buffer);
391987da915Sopenharmony_ci	return 0;
392987da915Sopenharmony_ci}
393987da915Sopenharmony_ci
394987da915Sopenharmony_ci/**
395987da915Sopenharmony_ci * main - Begin here
396987da915Sopenharmony_ci *
397987da915Sopenharmony_ci * Start from here.
398987da915Sopenharmony_ci *
399987da915Sopenharmony_ci * Return:  0  Success, the program worked
400987da915Sopenharmony_ci *	    1  Error, something went wrong
401987da915Sopenharmony_ci */
402987da915Sopenharmony_ciint main(int argc, char *argv[])
403987da915Sopenharmony_ci{
404987da915Sopenharmony_ci	ntfs_volume *vol;
405987da915Sopenharmony_ci	ntfs_inode *inode;
406987da915Sopenharmony_ci	ATTR_TYPES attr;
407987da915Sopenharmony_ci	int res;
408987da915Sopenharmony_ci	int result = 1;
409987da915Sopenharmony_ci
410987da915Sopenharmony_ci	ntfs_log_set_handler(ntfs_log_handler_stderr);
411987da915Sopenharmony_ci
412987da915Sopenharmony_ci	res = parse_options(argc, argv);
413987da915Sopenharmony_ci	if (res >= 0)
414987da915Sopenharmony_ci		return (res);
415987da915Sopenharmony_ci
416987da915Sopenharmony_ci	utils_set_locale();
417987da915Sopenharmony_ci
418987da915Sopenharmony_ci	vol = utils_mount_volume(opts.device, NTFS_MNT_RDONLY |
419987da915Sopenharmony_ci			(opts.force ? NTFS_MNT_RECOVER : 0));
420987da915Sopenharmony_ci	if (!vol) {
421987da915Sopenharmony_ci		ntfs_log_perror("ERROR: couldn't mount volume");
422987da915Sopenharmony_ci		return 1;
423987da915Sopenharmony_ci	}
424987da915Sopenharmony_ci
425987da915Sopenharmony_ci	if (opts.inode != -1)
426987da915Sopenharmony_ci		inode = ntfs_inode_open(vol, opts.inode);
427987da915Sopenharmony_ci	else {
428987da915Sopenharmony_ci#ifdef HAVE_WINDOWS_H
429987da915Sopenharmony_ci		char *unix_name;
430987da915Sopenharmony_ci
431987da915Sopenharmony_ci		unix_name = ntfs_utils_unix_path(opts.file);
432987da915Sopenharmony_ci		if (unix_name) {
433987da915Sopenharmony_ci			inode = ntfs_pathname_to_inode(vol, NULL,
434987da915Sopenharmony_ci					unix_name);
435987da915Sopenharmony_ci			free(unix_name);
436987da915Sopenharmony_ci		} else
437987da915Sopenharmony_ci			inode = (ntfs_inode*)NULL;
438987da915Sopenharmony_ci#else
439987da915Sopenharmony_ci		inode = ntfs_pathname_to_inode(vol, NULL, opts.file);
440987da915Sopenharmony_ci#endif
441987da915Sopenharmony_ci	}
442987da915Sopenharmony_ci
443987da915Sopenharmony_ci	if (!inode) {
444987da915Sopenharmony_ci		ntfs_log_perror("ERROR: Couldn't open inode");
445987da915Sopenharmony_ci		return 1;
446987da915Sopenharmony_ci	}
447987da915Sopenharmony_ci
448987da915Sopenharmony_ci	attr = AT_DATA;
449987da915Sopenharmony_ci	if (opts.attr != const_cpu_to_le32(-1))
450987da915Sopenharmony_ci		attr = opts.attr;
451987da915Sopenharmony_ci
452987da915Sopenharmony_ci	result = cat(vol, inode, attr, opts.attr_name, opts.attr_name_len);
453987da915Sopenharmony_ci
454987da915Sopenharmony_ci	ntfs_inode_close(inode);
455987da915Sopenharmony_ci	ntfs_umount(vol, FALSE);
456987da915Sopenharmony_ci
457987da915Sopenharmony_ci	return result;
458987da915Sopenharmony_ci}
459