1*9663SMark.Logan@Sun.COM /**
2*9663SMark.Logan@Sun.COM  * volume.c - NTFS volume handling code. Part of the Linux-NTFS project.
3*9663SMark.Logan@Sun.COM  *
4*9663SMark.Logan@Sun.COM  * Copyright (c) 2000-2006 Anton Altaparmakov
5*9663SMark.Logan@Sun.COM  * Copyright (c) 2002-2006 Szabolcs Szakacsits
6*9663SMark.Logan@Sun.COM  * Copyright (c) 2004-2005 Richard Russon
7*9663SMark.Logan@Sun.COM  * Copyright (c) 2005-2007 Yura Pakhuchiy
8*9663SMark.Logan@Sun.COM  *
9*9663SMark.Logan@Sun.COM  * This program/include file is free software; you can redistribute it and/or
10*9663SMark.Logan@Sun.COM  * modify it under the terms of the GNU General Public License as published
11*9663SMark.Logan@Sun.COM  * by the Free Software Foundation; either version 2 of the License, or
12*9663SMark.Logan@Sun.COM  * (at your option) any later version.
13*9663SMark.Logan@Sun.COM  *
14*9663SMark.Logan@Sun.COM  * This program/include file is distributed in the hope that it will be
15*9663SMark.Logan@Sun.COM  * useful, but WITHOUT ANY WARRANTY; without even the implied warranty
16*9663SMark.Logan@Sun.COM  * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17*9663SMark.Logan@Sun.COM  * GNU General Public License for more details.
18*9663SMark.Logan@Sun.COM  *
19*9663SMark.Logan@Sun.COM  * You should have received a copy of the GNU General Public License
20*9663SMark.Logan@Sun.COM  * along with this program (in the main directory of the Linux-NTFS
21*9663SMark.Logan@Sun.COM  * distribution in the file COPYING); if not, write to the Free Software
22*9663SMark.Logan@Sun.COM  * Foundation,Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
23*9663SMark.Logan@Sun.COM  */
24*9663SMark.Logan@Sun.COM 
25*9663SMark.Logan@Sun.COM #ifdef HAVE_CONFIG_H
26*9663SMark.Logan@Sun.COM #include "config.h"
27*9663SMark.Logan@Sun.COM #endif
28*9663SMark.Logan@Sun.COM 
29*9663SMark.Logan@Sun.COM #ifdef HAVE_STDLIB_H
30*9663SMark.Logan@Sun.COM #include <stdlib.h>
31*9663SMark.Logan@Sun.COM #endif
32*9663SMark.Logan@Sun.COM #ifdef HAVE_STDIO_H
33*9663SMark.Logan@Sun.COM #include <stdio.h>
34*9663SMark.Logan@Sun.COM #endif
35*9663SMark.Logan@Sun.COM #ifdef HAVE_STRING_H
36*9663SMark.Logan@Sun.COM #include <string.h>
37*9663SMark.Logan@Sun.COM #endif
38*9663SMark.Logan@Sun.COM #ifdef HAVE_FCNTL_H
39*9663SMark.Logan@Sun.COM #include <fcntl.h>
40*9663SMark.Logan@Sun.COM #endif
41*9663SMark.Logan@Sun.COM #ifdef HAVE_UNISTD_H
42*9663SMark.Logan@Sun.COM #include <unistd.h>
43*9663SMark.Logan@Sun.COM #endif
44*9663SMark.Logan@Sun.COM #ifdef HAVE_ERRNO_H
45*9663SMark.Logan@Sun.COM #include <errno.h>
46*9663SMark.Logan@Sun.COM #endif
47*9663SMark.Logan@Sun.COM #ifdef HAVE_SYS_STAT_H
48*9663SMark.Logan@Sun.COM #include <sys/stat.h>
49*9663SMark.Logan@Sun.COM #endif
50*9663SMark.Logan@Sun.COM #ifdef HAVE_LIMITS_H
51*9663SMark.Logan@Sun.COM #include <limits.h>
52*9663SMark.Logan@Sun.COM #endif
53*9663SMark.Logan@Sun.COM 
54*9663SMark.Logan@Sun.COM #include "volume.h"
55*9663SMark.Logan@Sun.COM #include "attrib.h"
56*9663SMark.Logan@Sun.COM #include "mft.h"
57*9663SMark.Logan@Sun.COM #include "bootsect.h"
58*9663SMark.Logan@Sun.COM #include "device.h"
59*9663SMark.Logan@Sun.COM #include "debug.h"
60*9663SMark.Logan@Sun.COM #include "inode.h"
61*9663SMark.Logan@Sun.COM #include "runlist.h"
62*9663SMark.Logan@Sun.COM #include "logfile.h"
63*9663SMark.Logan@Sun.COM #include "dir.h"
64*9663SMark.Logan@Sun.COM #include "logging.h"
65*9663SMark.Logan@Sun.COM 
66*9663SMark.Logan@Sun.COM #ifndef PATH_MAX
67*9663SMark.Logan@Sun.COM #define PATH_MAX 4096
68*9663SMark.Logan@Sun.COM #endif
69*9663SMark.Logan@Sun.COM 
70*9663SMark.Logan@Sun.COM /**
71*9663SMark.Logan@Sun.COM  * ntfs_volume_alloc - Create an NTFS volume object and initialise it
72*9663SMark.Logan@Sun.COM  *
73*9663SMark.Logan@Sun.COM  * Description...
74*9663SMark.Logan@Sun.COM  *
75*9663SMark.Logan@Sun.COM  * Returns:
76*9663SMark.Logan@Sun.COM  */
77*9663SMark.Logan@Sun.COM ntfs_volume *ntfs_volume_alloc(void)
78*9663SMark.Logan@Sun.COM {
79*9663SMark.Logan@Sun.COM 	ntfs_volume *vol;
80*9663SMark.Logan@Sun.COM 	int i;
81*9663SMark.Logan@Sun.COM 
82*9663SMark.Logan@Sun.COM 	vol = calloc(1, sizeof(ntfs_volume));
83*9663SMark.Logan@Sun.COM 	if (vol) {
84*9663SMark.Logan@Sun.COM 		for (i = 0; i < NTFS_INODE_CACHE_SIZE; i++)
85*9663SMark.Logan@Sun.COM 			INIT_LIST_HEAD(&vol->inode_cache[i]);
86*9663SMark.Logan@Sun.COM 	}
87*9663SMark.Logan@Sun.COM 	return vol;
88*9663SMark.Logan@Sun.COM }
89*9663SMark.Logan@Sun.COM 
90*9663SMark.Logan@Sun.COM /**
91*9663SMark.Logan@Sun.COM  * __ntfs_volume_release - Destroy an NTFS volume object
92*9663SMark.Logan@Sun.COM  * @v:
93*9663SMark.Logan@Sun.COM  *
94*9663SMark.Logan@Sun.COM  * Description...
95*9663SMark.Logan@Sun.COM  *
96*9663SMark.Logan@Sun.COM  * Returns:
97*9663SMark.Logan@Sun.COM  */
98*9663SMark.Logan@Sun.COM static void __ntfs_volume_release(ntfs_volume *v)
99*9663SMark.Logan@Sun.COM {
100*9663SMark.Logan@Sun.COM 	struct list_head *pos, *tmp;
101*9663SMark.Logan@Sun.COM 	int i;
102*9663SMark.Logan@Sun.COM 
103*9663SMark.Logan@Sun.COM 	/* Sync and print error about not detached inodes. */
104*9663SMark.Logan@Sun.COM 	for (i = 0; i < NTFS_INODE_CACHE_SIZE; i++)
105*9663SMark.Logan@Sun.COM 		list_for_each_safe(pos, tmp, &v->inode_cache[i]) {
106*9663SMark.Logan@Sun.COM 			ntfs_inode *ni =
107*9663SMark.Logan@Sun.COM 					list_entry(pos, ntfs_inode, list_entry);
108*9663SMark.Logan@Sun.COM 
109*9663SMark.Logan@Sun.COM 			switch (ni->mft_no) {
110*9663SMark.Logan@Sun.COM 				case FILE_Volume:
111*9663SMark.Logan@Sun.COM 				case FILE_Bitmap:
112*9663SMark.Logan@Sun.COM 				case FILE_MFT:
113*9663SMark.Logan@Sun.COM 				case FILE_MFTMirr:
114*9663SMark.Logan@Sun.COM 					if (ni->nr_references == 1)
115*9663SMark.Logan@Sun.COM 						continue;
116*9663SMark.Logan@Sun.COM 					break;
117*9663SMark.Logan@Sun.COM 			}
118*9663SMark.Logan@Sun.COM 
119*9663SMark.Logan@Sun.COM 			ntfs_log_error("%s(): Inode %llu still have %d "
120*9663SMark.Logan@Sun.COM 					"references.\n", "__ntfs_volume_release",
121*9663SMark.Logan@Sun.COM 					ni->mft_no, ni->nr_references);
122*9663SMark.Logan@Sun.COM 			ntfs_inode_sync(ni);
123*9663SMark.Logan@Sun.COM 		}
124*9663SMark.Logan@Sun.COM 	/*
125*9663SMark.Logan@Sun.COM 	 * Clear the dirty bit if it was not set before we mounted and this is
126*9663SMark.Logan@Sun.COM 	 * not a forensic mount.
127*9663SMark.Logan@Sun.COM 	 */
128*9663SMark.Logan@Sun.COM 	if (!NVolReadOnly(v) && !NVolWasDirty(v) && !NVolForensicMount(v)) {
129*9663SMark.Logan@Sun.COM 		v->flags &= ~VOLUME_IS_DIRTY;
130*9663SMark.Logan@Sun.COM 		(void)ntfs_volume_write_flags(v, v->flags);
131*9663SMark.Logan@Sun.COM 	}
132*9663SMark.Logan@Sun.COM 	if (v->lcnbmp_ni && NInoDirty(v->lcnbmp_ni))
133*9663SMark.Logan@Sun.COM 		ntfs_inode_sync(v->lcnbmp_ni);
134*9663SMark.Logan@Sun.COM 	if (v->vol_ni)
135*9663SMark.Logan@Sun.COM 		ntfs_inode_close(v->vol_ni);
136*9663SMark.Logan@Sun.COM 	if (v->lcnbmp_na)
137*9663SMark.Logan@Sun.COM 		ntfs_attr_close(v->lcnbmp_na);
138*9663SMark.Logan@Sun.COM 	if (v->lcnbmp_ni)
139*9663SMark.Logan@Sun.COM 		ntfs_inode_close(v->lcnbmp_ni);
140*9663SMark.Logan@Sun.COM 	if (v->mft_ni && NInoDirty(v->mft_ni))
141*9663SMark.Logan@Sun.COM 		ntfs_inode_sync(v->mft_ni);
142*9663SMark.Logan@Sun.COM 	if (v->mftbmp_na)
143*9663SMark.Logan@Sun.COM 		ntfs_attr_close(v->mftbmp_na);
144*9663SMark.Logan@Sun.COM 	if (v->mft_na)
145*9663SMark.Logan@Sun.COM 		ntfs_attr_close(v->mft_na);
146*9663SMark.Logan@Sun.COM 	if (v->mft_ni)
147*9663SMark.Logan@Sun.COM 		ntfs_inode_close(v->mft_ni);
148*9663SMark.Logan@Sun.COM 	if (v->mftmirr_ni && NInoDirty(v->mftmirr_ni))
149*9663SMark.Logan@Sun.COM 		ntfs_inode_sync(v->mftmirr_ni);
150*9663SMark.Logan@Sun.COM 	if (v->mftmirr_na)
151*9663SMark.Logan@Sun.COM 		ntfs_attr_close(v->mftmirr_na);
152*9663SMark.Logan@Sun.COM 	if (v->mftmirr_ni)
153*9663SMark.Logan@Sun.COM 		ntfs_inode_close(v->mftmirr_ni);
154*9663SMark.Logan@Sun.COM 	if (v->u.dev) {
155*9663SMark.Logan@Sun.COM 		struct ntfs_device *dev = v->u.dev;
156*9663SMark.Logan@Sun.COM 
157*9663SMark.Logan@Sun.COM 		if (NDevDirty(dev))
158*9663SMark.Logan@Sun.COM 			dev->d_ops->sync(dev);
159*9663SMark.Logan@Sun.COM 		if (dev->d_ops->close(dev))
160*9663SMark.Logan@Sun.COM 			ntfs_log_perror("Failed to close the device");
161*9663SMark.Logan@Sun.COM 	}
162*9663SMark.Logan@Sun.COM 	free(v->vol_name);
163*9663SMark.Logan@Sun.COM 	free(v->upcase);
164*9663SMark.Logan@Sun.COM 	free(v->attrdef);
165*9663SMark.Logan@Sun.COM 	free(v);
166*9663SMark.Logan@Sun.COM }
167*9663SMark.Logan@Sun.COM 
168*9663SMark.Logan@Sun.COM /**
169*9663SMark.Logan@Sun.COM  * ntfs_mft_load - load the $MFT and setup the ntfs volume with it
170*9663SMark.Logan@Sun.COM  * @vol:	ntfs volume whose $MFT to load
171*9663SMark.Logan@Sun.COM  *
172*9663SMark.Logan@Sun.COM  * Load $MFT from @vol and setup @vol with it. After calling this function the
173*9663SMark.Logan@Sun.COM  * volume @vol is ready for use by all read access functions provided by the
174*9663SMark.Logan@Sun.COM  * ntfs library.
175*9663SMark.Logan@Sun.COM  *
176*9663SMark.Logan@Sun.COM  * Return 0 on success and -1 on error with errno set to the error code.
177*9663SMark.Logan@Sun.COM  */
178*9663SMark.Logan@Sun.COM static int ntfs_mft_load(ntfs_volume *vol)
179*9663SMark.Logan@Sun.COM {
180*9663SMark.Logan@Sun.COM 	VCN next_vcn, last_vcn, highest_vcn;
181*9663SMark.Logan@Sun.COM 	s64 l;
182*9663SMark.Logan@Sun.COM 	MFT_RECORD *mb = NULL;
183*9663SMark.Logan@Sun.COM 	ntfs_attr_search_ctx *ctx = NULL;
184*9663SMark.Logan@Sun.COM 	ATTR_RECORD *a;
185*9663SMark.Logan@Sun.COM 	STANDARD_INFORMATION *std_info;
186*9663SMark.Logan@Sun.COM 	int eo;
187*9663SMark.Logan@Sun.COM 
188*9663SMark.Logan@Sun.COM 	/* Manually setup an ntfs_inode. */
189*9663SMark.Logan@Sun.COM 	vol->mft_ni = ntfs_inode_allocate(vol);
190*9663SMark.Logan@Sun.COM 	mb = (MFT_RECORD*)ntfs_malloc(vol->mft_record_size);
191*9663SMark.Logan@Sun.COM 	if (!vol->mft_ni || !mb) {
192*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Error allocating memory for $MFT");
193*9663SMark.Logan@Sun.COM 		goto error_exit;
194*9663SMark.Logan@Sun.COM 	}
195*9663SMark.Logan@Sun.COM 	vol->mft_ni->mft_no = 0;
196*9663SMark.Logan@Sun.COM 	vol->mft_ni->mrec = mb;
197*9663SMark.Logan@Sun.COM 	__ntfs_inode_add_to_cache(vol->mft_ni);
198*9663SMark.Logan@Sun.COM 	/* Can't use any of the higher level functions yet! */
199*9663SMark.Logan@Sun.COM 	l = ntfs_mst_pread(vol->u.dev, vol->mft_lcn << vol->cluster_size_bits, 1,
200*9663SMark.Logan@Sun.COM 			vol->mft_record_size, mb);
201*9663SMark.Logan@Sun.COM 	if (l != 1) {
202*9663SMark.Logan@Sun.COM 		if (l != -1)
203*9663SMark.Logan@Sun.COM 			errno = EIO;
204*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Error reading $MFT");
205*9663SMark.Logan@Sun.COM 		goto error_exit;
206*9663SMark.Logan@Sun.COM 	}
207*9663SMark.Logan@Sun.COM 	if (ntfs_is_baad_record(mb->magic)) {
208*9663SMark.Logan@Sun.COM 		ntfs_log_error("Incomplete multi sector transfer detected in "
209*9663SMark.Logan@Sun.COM 				"$MFT.\n");
210*9663SMark.Logan@Sun.COM 		goto io_error_exit;
211*9663SMark.Logan@Sun.COM 	}
212*9663SMark.Logan@Sun.COM 	if (!ntfs_is_mft_record(mb->magic)) {
213*9663SMark.Logan@Sun.COM 		ntfs_log_error("$MFT has invalid magic.\n");
214*9663SMark.Logan@Sun.COM 		goto io_error_exit;
215*9663SMark.Logan@Sun.COM 	}
216*9663SMark.Logan@Sun.COM 	ctx = ntfs_attr_get_search_ctx(vol->mft_ni, NULL);
217*9663SMark.Logan@Sun.COM 	if (!ctx) {
218*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to allocate attribute search context");
219*9663SMark.Logan@Sun.COM 		goto error_exit;
220*9663SMark.Logan@Sun.COM 	}
221*9663SMark.Logan@Sun.COM 	if (p2n(ctx->attr) < p2n(mb) ||
222*9663SMark.Logan@Sun.COM 			(char*)ctx->attr > (char*)mb + vol->mft_record_size) {
223*9663SMark.Logan@Sun.COM 		ntfs_log_error("$MFT is corrupt.\n");
224*9663SMark.Logan@Sun.COM 		goto io_error_exit;
225*9663SMark.Logan@Sun.COM 	}
226*9663SMark.Logan@Sun.COM 	/* Find the $ATTRIBUTE_LIST attribute in $MFT if present. */
227*9663SMark.Logan@Sun.COM 	if (ntfs_attr_lookup(AT_ATTRIBUTE_LIST, AT_UNNAMED, 0, 0, 0, NULL, 0,
228*9663SMark.Logan@Sun.COM 			ctx)) {
229*9663SMark.Logan@Sun.COM 		if (errno != ENOENT) {
230*9663SMark.Logan@Sun.COM 			ntfs_log_error("$MFT has corrupt attribute list.\n");
231*9663SMark.Logan@Sun.COM 			goto io_error_exit;
232*9663SMark.Logan@Sun.COM 		}
233*9663SMark.Logan@Sun.COM 		goto mft_has_no_attr_list;
234*9663SMark.Logan@Sun.COM 	}
235*9663SMark.Logan@Sun.COM 	NInoSetAttrList(vol->mft_ni);
236*9663SMark.Logan@Sun.COM 	l = ntfs_get_attribute_value_length(ctx->attr);
237*9663SMark.Logan@Sun.COM 	if (l <= 0 || l > 0x40000) {
238*9663SMark.Logan@Sun.COM 		ntfs_log_error("$MFT/$ATTRIBUTE_LIST has invalid length.\n");
239*9663SMark.Logan@Sun.COM 		goto io_error_exit;
240*9663SMark.Logan@Sun.COM 	}
241*9663SMark.Logan@Sun.COM 	vol->mft_ni->attr_list_size = l;
242*9663SMark.Logan@Sun.COM 	vol->mft_ni->attr_list = ntfs_malloc(l);
243*9663SMark.Logan@Sun.COM 	if (!vol->mft_ni->attr_list)
244*9663SMark.Logan@Sun.COM 		goto error_exit;
245*9663SMark.Logan@Sun.COM 
246*9663SMark.Logan@Sun.COM 	l = ntfs_get_attribute_value(vol, ctx->attr, vol->mft_ni->attr_list);
247*9663SMark.Logan@Sun.COM 	if (!l) {
248*9663SMark.Logan@Sun.COM 		ntfs_log_error("Failed to get value of "
249*9663SMark.Logan@Sun.COM 				"$MFT/$ATTRIBUTE_LIST.\n");
250*9663SMark.Logan@Sun.COM 		goto io_error_exit;
251*9663SMark.Logan@Sun.COM 	}
252*9663SMark.Logan@Sun.COM 	if (l != vol->mft_ni->attr_list_size) {
253*9663SMark.Logan@Sun.COM 		ntfs_log_error("Got unexpected amount of data when "
254*9663SMark.Logan@Sun.COM 				"reading $MFT/$ATTRIBUTE_LIST.\n");
255*9663SMark.Logan@Sun.COM 		goto io_error_exit;
256*9663SMark.Logan@Sun.COM 	}
257*9663SMark.Logan@Sun.COM mft_has_no_attr_list:
258*9663SMark.Logan@Sun.COM 	/* Receive attributes from STANDARD_INFORMATION. */
259*9663SMark.Logan@Sun.COM 	std_info = ntfs_attr_readall(vol->mft_ni, AT_STANDARD_INFORMATION,
260*9663SMark.Logan@Sun.COM 				     AT_UNNAMED, 0, NULL);
261*9663SMark.Logan@Sun.COM 	vol->mft_ni->flags = std_info->file_attributes;
262*9663SMark.Logan@Sun.COM 	free(std_info);
263*9663SMark.Logan@Sun.COM 
264*9663SMark.Logan@Sun.COM 	/* We now have a fully setup ntfs inode for $MFT in vol->mft_ni. */
265*9663SMark.Logan@Sun.COM 
266*9663SMark.Logan@Sun.COM 	/* Get an ntfs attribute for $MFT/$DATA and set it up, too. */
267*9663SMark.Logan@Sun.COM 	vol->mft_na = ntfs_attr_open(vol->mft_ni, AT_DATA, AT_UNNAMED, 0);
268*9663SMark.Logan@Sun.COM 	if (!vol->mft_na) {
269*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open ntfs attribute");
270*9663SMark.Logan@Sun.COM 		goto error_exit;
271*9663SMark.Logan@Sun.COM 	}
272*9663SMark.Logan@Sun.COM 	/* Read all extents from the $DATA attribute in $MFT. */
273*9663SMark.Logan@Sun.COM 	ntfs_attr_reinit_search_ctx(ctx);
274*9663SMark.Logan@Sun.COM 	last_vcn = vol->mft_na->allocated_size >> vol->cluster_size_bits;
275*9663SMark.Logan@Sun.COM 	highest_vcn = next_vcn = 0;
276*9663SMark.Logan@Sun.COM 	a = NULL;
277*9663SMark.Logan@Sun.COM 	while (!ntfs_attr_lookup(AT_DATA, AT_UNNAMED, 0, 0, next_vcn, NULL, 0,
278*9663SMark.Logan@Sun.COM 			ctx)) {
279*9663SMark.Logan@Sun.COM 		runlist_element *nrl;
280*9663SMark.Logan@Sun.COM 
281*9663SMark.Logan@Sun.COM 		a = ctx->attr;
282*9663SMark.Logan@Sun.COM 		/* $MFT must be non-resident. */
283*9663SMark.Logan@Sun.COM 		if (!a->non_resident) {
284*9663SMark.Logan@Sun.COM 			ntfs_log_error("$MFT must be non-resident but a "
285*9663SMark.Logan@Sun.COM 					"resident extent was found. $MFT is "
286*9663SMark.Logan@Sun.COM 					"corrupt. Run chkdsk.\n");
287*9663SMark.Logan@Sun.COM 			goto io_error_exit;
288*9663SMark.Logan@Sun.COM 		}
289*9663SMark.Logan@Sun.COM 		/* $MFT must be uncompressed and unencrypted. */
290*9663SMark.Logan@Sun.COM 		if (a->flags & ATTR_COMPRESSION_MASK ||
291*9663SMark.Logan@Sun.COM 				a->flags & ATTR_IS_ENCRYPTED) {
292*9663SMark.Logan@Sun.COM 			ntfs_log_error("$MFT must be uncompressed and "
293*9663SMark.Logan@Sun.COM 					"unencrypted but a compressed/encrypted"
294*9663SMark.Logan@Sun.COM 					" extent was found. $MFT is corrupt. "
295*9663SMark.Logan@Sun.COM 					"Run chkdsk.\n");
296*9663SMark.Logan@Sun.COM 			goto io_error_exit;
297*9663SMark.Logan@Sun.COM 		}
298*9663SMark.Logan@Sun.COM 		/*
299*9663SMark.Logan@Sun.COM 		 * Decompress the mapping pairs array of this extent and merge
300*9663SMark.Logan@Sun.COM 		 * the result into the existing runlist. No need for locking
301*9663SMark.Logan@Sun.COM 		 * as we have exclusive access to the inode at this time and we
302*9663SMark.Logan@Sun.COM 		 * are a mount in progress task, too.
303*9663SMark.Logan@Sun.COM 		 */
304*9663SMark.Logan@Sun.COM 		nrl = ntfs_mapping_pairs_decompress(vol, a, vol->mft_na->rl);
305*9663SMark.Logan@Sun.COM 		if (!nrl) {
306*9663SMark.Logan@Sun.COM 			ntfs_log_perror("ntfs_mapping_pairs_decompress() "
307*9663SMark.Logan@Sun.COM 					"failed");
308*9663SMark.Logan@Sun.COM 			goto error_exit;
309*9663SMark.Logan@Sun.COM 		}
310*9663SMark.Logan@Sun.COM 		vol->mft_na->rl = nrl;
311*9663SMark.Logan@Sun.COM 
312*9663SMark.Logan@Sun.COM 		/* Get the lowest vcn for the next extent. */
313*9663SMark.Logan@Sun.COM 		highest_vcn = sle64_to_cpu(a->u.nonres.highest_vcn);
314*9663SMark.Logan@Sun.COM 		next_vcn = highest_vcn + 1;
315*9663SMark.Logan@Sun.COM 
316*9663SMark.Logan@Sun.COM 		/* Only one extent or error, which we catch below. */
317*9663SMark.Logan@Sun.COM 		if (next_vcn <= 0)
318*9663SMark.Logan@Sun.COM 			break;
319*9663SMark.Logan@Sun.COM 
320*9663SMark.Logan@Sun.COM 		/* Avoid endless loops due to corruption. */
321*9663SMark.Logan@Sun.COM 		if (next_vcn < sle64_to_cpu(a->u.nonres.lowest_vcn)) {
322*9663SMark.Logan@Sun.COM 			ntfs_log_error("$MFT has corrupt attribute list "
323*9663SMark.Logan@Sun.COM 					"attribute. Run chkdsk.\n");
324*9663SMark.Logan@Sun.COM 			goto io_error_exit;
325*9663SMark.Logan@Sun.COM 		}
326*9663SMark.Logan@Sun.COM 	}
327*9663SMark.Logan@Sun.COM 	if (!a) {
328*9663SMark.Logan@Sun.COM 		ntfs_log_error("$MFT/$DATA attribute not found. "
329*9663SMark.Logan@Sun.COM 				"$MFT is corrupt. Run chkdsk.\n");
330*9663SMark.Logan@Sun.COM 		goto io_error_exit;
331*9663SMark.Logan@Sun.COM 	}
332*9663SMark.Logan@Sun.COM 	if (highest_vcn && highest_vcn != last_vcn - 1) {
333*9663SMark.Logan@Sun.COM 		ntfs_log_error("Failed to load the complete runlist for "
334*9663SMark.Logan@Sun.COM 				"$MFT/$DATA. Bug or corrupt $MFT. "
335*9663SMark.Logan@Sun.COM 				"Run chkdsk.\n highest_vcn = 0x%llx, "
336*9663SMark.Logan@Sun.COM 				"last_vcn - 1 = 0x%llx\n", (long long)
337*9663SMark.Logan@Sun.COM 				highest_vcn, (long long)last_vcn - 1);
338*9663SMark.Logan@Sun.COM 		goto io_error_exit;
339*9663SMark.Logan@Sun.COM 	}
340*9663SMark.Logan@Sun.COM 	/* Done with the $Mft mft record. */
341*9663SMark.Logan@Sun.COM 	ntfs_attr_put_search_ctx(ctx);
342*9663SMark.Logan@Sun.COM 	ctx = NULL;
343*9663SMark.Logan@Sun.COM 	/*
344*9663SMark.Logan@Sun.COM 	 * The volume is now setup so we can use all read access functions.
345*9663SMark.Logan@Sun.COM 	 */
346*9663SMark.Logan@Sun.COM 	vol->mftbmp_na = ntfs_attr_open(vol->mft_ni, AT_BITMAP, AT_UNNAMED, 0);
347*9663SMark.Logan@Sun.COM 	if (!vol->mftbmp_na) {
348*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open $MFT/$BITMAP");
349*9663SMark.Logan@Sun.COM 		goto error_exit;
350*9663SMark.Logan@Sun.COM 	}
351*9663SMark.Logan@Sun.COM 	return 0;
352*9663SMark.Logan@Sun.COM io_error_exit:
353*9663SMark.Logan@Sun.COM 	errno = EIO;
354*9663SMark.Logan@Sun.COM error_exit:
355*9663SMark.Logan@Sun.COM 	eo = errno;
356*9663SMark.Logan@Sun.COM 	if (ctx)
357*9663SMark.Logan@Sun.COM 		ntfs_attr_put_search_ctx(ctx);
358*9663SMark.Logan@Sun.COM 	if (vol->mft_na) {
359*9663SMark.Logan@Sun.COM 		ntfs_attr_close(vol->mft_na);
360*9663SMark.Logan@Sun.COM 		vol->mft_na = NULL;
361*9663SMark.Logan@Sun.COM 	}
362*9663SMark.Logan@Sun.COM 	if (vol->mft_ni) {
363*9663SMark.Logan@Sun.COM 		ntfs_inode_close(vol->mft_ni);
364*9663SMark.Logan@Sun.COM 		vol->mft_ni = NULL;
365*9663SMark.Logan@Sun.COM 	}
366*9663SMark.Logan@Sun.COM 	ntfs_log_error("%s(): Failed.\n", "ntfs_mft_load");
367*9663SMark.Logan@Sun.COM 	errno = eo;
368*9663SMark.Logan@Sun.COM 	return -1;
369*9663SMark.Logan@Sun.COM }
370*9663SMark.Logan@Sun.COM 
371*9663SMark.Logan@Sun.COM /**
372*9663SMark.Logan@Sun.COM  * ntfs_mftmirr_load - load the $MFTMirr and setup the ntfs volume with it
373*9663SMark.Logan@Sun.COM  * @vol:	ntfs volume whose $MFTMirr to load
374*9663SMark.Logan@Sun.COM  *
375*9663SMark.Logan@Sun.COM  * Load $MFTMirr from @vol and setup @vol with it. After calling this function
376*9663SMark.Logan@Sun.COM  * the volume @vol is ready for use by all write access functions provided by
377*9663SMark.Logan@Sun.COM  * the ntfs library (assuming ntfs_mft_load() has been called successfully
378*9663SMark.Logan@Sun.COM  * beforehand).
379*9663SMark.Logan@Sun.COM  *
380*9663SMark.Logan@Sun.COM  * Return 0 on success and -1 on error with errno set to the error code.
381*9663SMark.Logan@Sun.COM  */
382*9663SMark.Logan@Sun.COM static int ntfs_mftmirr_load(ntfs_volume *vol)
383*9663SMark.Logan@Sun.COM {
384*9663SMark.Logan@Sun.COM 	int err;
385*9663SMark.Logan@Sun.COM 
386*9663SMark.Logan@Sun.COM 	vol->mftmirr_ni = ntfs_inode_open(vol, FILE_MFTMirr);
387*9663SMark.Logan@Sun.COM 	if (!vol->mftmirr_ni) {
388*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open inode $MFTMirr");
389*9663SMark.Logan@Sun.COM 		return -1;
390*9663SMark.Logan@Sun.COM 	}
391*9663SMark.Logan@Sun.COM 	/* Get an ntfs attribute for $MFTMirr/$DATA, too. */
392*9663SMark.Logan@Sun.COM 	vol->mftmirr_na = ntfs_attr_open(vol->mftmirr_ni, AT_DATA,
393*9663SMark.Logan@Sun.COM 			AT_UNNAMED, 0);
394*9663SMark.Logan@Sun.COM 	if (!vol->mftmirr_na) {
395*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open $MFTMirr/$DATA");
396*9663SMark.Logan@Sun.COM 		goto error_exit;
397*9663SMark.Logan@Sun.COM 	}
398*9663SMark.Logan@Sun.COM 	if (ntfs_attr_map_runlist(vol->mftmirr_na, 0) < 0) {
399*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to map runlist of $MFTMirr/$DATA");
400*9663SMark.Logan@Sun.COM 		goto error_exit;
401*9663SMark.Logan@Sun.COM 	}
402*9663SMark.Logan@Sun.COM 	/* Check $MFTMirr runlist. */
403*9663SMark.Logan@Sun.COM 	if (vol->mftmirr_na->rl[0].lcn != vol->mftmirr_lcn ||
404*9663SMark.Logan@Sun.COM 			vol->mftmirr_na->rl[0].length < (vol->mftmirr_size *
405*9663SMark.Logan@Sun.COM 			vol->mft_record_size + vol->cluster_size - 1) /
406*9663SMark.Logan@Sun.COM 			vol->cluster_size) {
407*9663SMark.Logan@Sun.COM 		ntfs_log_error("$MFTMirr location mismatch or first 4 records "
408*9663SMark.Logan@Sun.COM 				"are fragmented. Run chkdsk.\n");
409*9663SMark.Logan@Sun.COM 		errno = EIO;
410*9663SMark.Logan@Sun.COM 		goto error_exit;
411*9663SMark.Logan@Sun.COM 
412*9663SMark.Logan@Sun.COM 	}
413*9663SMark.Logan@Sun.COM 	return 0;
414*9663SMark.Logan@Sun.COM error_exit:
415*9663SMark.Logan@Sun.COM 	err = errno;
416*9663SMark.Logan@Sun.COM 	if (vol->mftmirr_na) {
417*9663SMark.Logan@Sun.COM 		ntfs_attr_close(vol->mftmirr_na);
418*9663SMark.Logan@Sun.COM 		vol->mftmirr_na = NULL;
419*9663SMark.Logan@Sun.COM 	}
420*9663SMark.Logan@Sun.COM 	ntfs_inode_close(vol->mftmirr_ni);
421*9663SMark.Logan@Sun.COM 	vol->mftmirr_ni = NULL;
422*9663SMark.Logan@Sun.COM 	errno = err;
423*9663SMark.Logan@Sun.COM 	return -1;
424*9663SMark.Logan@Sun.COM }
425*9663SMark.Logan@Sun.COM 
426*9663SMark.Logan@Sun.COM /**
427*9663SMark.Logan@Sun.COM  * ntfs_volume_startup - allocate and setup an ntfs volume
428*9663SMark.Logan@Sun.COM  * @dev:	device to open
429*9663SMark.Logan@Sun.COM  * @flags:	optional mount flags
430*9663SMark.Logan@Sun.COM  *
431*9663SMark.Logan@Sun.COM  * Load, verify, and parse bootsector; load and setup $MFT and $MFTMirr. After
432*9663SMark.Logan@Sun.COM  * calling this function, the volume is setup sufficiently to call all read
433*9663SMark.Logan@Sun.COM  * and write access functions provided by the library.
434*9663SMark.Logan@Sun.COM  *
435*9663SMark.Logan@Sun.COM  * Return the allocated volume structure on success and NULL on error with
436*9663SMark.Logan@Sun.COM  * errno set to the error code.
437*9663SMark.Logan@Sun.COM  */
438*9663SMark.Logan@Sun.COM ntfs_volume *ntfs_volume_startup(struct ntfs_device *dev,
439*9663SMark.Logan@Sun.COM 		ntfs_mount_flags flags)
440*9663SMark.Logan@Sun.COM {
441*9663SMark.Logan@Sun.COM 	LCN mft_zone_size, mft_lcn;
442*9663SMark.Logan@Sun.COM 	s64 br;
443*9663SMark.Logan@Sun.COM 	ntfs_volume *vol;
444*9663SMark.Logan@Sun.COM 	NTFS_BOOT_SECTOR *bs;
445*9663SMark.Logan@Sun.COM 	int eo;
446*9663SMark.Logan@Sun.COM #ifdef DEBUG
447*9663SMark.Logan@Sun.COM 	const char *OK = "OK\n";
448*9663SMark.Logan@Sun.COM 	const char *FAILED = "FAILED\n";
449*9663SMark.Logan@Sun.COM 	BOOL debug = 1;
450*9663SMark.Logan@Sun.COM #else
451*9663SMark.Logan@Sun.COM 	BOOL debug = 0;
452*9663SMark.Logan@Sun.COM #endif
453*9663SMark.Logan@Sun.COM 
454*9663SMark.Logan@Sun.COM 	if (!dev || !dev->d_ops || !dev->d_name) {
455*9663SMark.Logan@Sun.COM 		errno = EINVAL;
456*9663SMark.Logan@Sun.COM 		return NULL;
457*9663SMark.Logan@Sun.COM 	}
458*9663SMark.Logan@Sun.COM 
459*9663SMark.Logan@Sun.COM 	if (!(bs = (NTFS_BOOT_SECTOR *)ntfs_malloc(sizeof(NTFS_BOOT_SECTOR))))
460*9663SMark.Logan@Sun.COM 		return NULL;
461*9663SMark.Logan@Sun.COM 
462*9663SMark.Logan@Sun.COM 	/* Allocate the volume structure. */
463*9663SMark.Logan@Sun.COM 	vol = ntfs_volume_alloc();
464*9663SMark.Logan@Sun.COM 	if (!vol)
465*9663SMark.Logan@Sun.COM 		goto error_exit;
466*9663SMark.Logan@Sun.COM 	/* Create the default upcase table. */
467*9663SMark.Logan@Sun.COM 	vol->upcase_len = 65536;
468*9663SMark.Logan@Sun.COM 	vol->upcase = (ntfschar*)ntfs_malloc(vol->upcase_len *
469*9663SMark.Logan@Sun.COM 			sizeof(ntfschar));
470*9663SMark.Logan@Sun.COM 	if (!vol->upcase)
471*9663SMark.Logan@Sun.COM 		goto error_exit;
472*9663SMark.Logan@Sun.COM 	ntfs_upcase_table_build(vol->upcase,
473*9663SMark.Logan@Sun.COM 			vol->upcase_len * sizeof(ntfschar));
474*9663SMark.Logan@Sun.COM 	if (flags & NTFS_MNT_RDONLY)
475*9663SMark.Logan@Sun.COM 		NVolSetReadOnly(vol);
476*9663SMark.Logan@Sun.COM 	if (flags & NTFS_MNT_CASE_SENSITIVE)
477*9663SMark.Logan@Sun.COM 		NVolSetCaseSensitive(vol);
478*9663SMark.Logan@Sun.COM 	if (flags & NTFS_MNT_INTERIX)
479*9663SMark.Logan@Sun.COM 		NVolSetInterix(vol);
480*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Reading bootsector... ");
481*9663SMark.Logan@Sun.COM 	if (dev->d_ops->open(dev, NVolReadOnly(vol) ? O_RDONLY :
482*9663SMark.Logan@Sun.COM 				((flags & NTFS_MNT_NOT_EXCLUSIVE) ? O_RDWR :
483*9663SMark.Logan@Sun.COM 				(O_RDWR | O_EXCL)))) {
484*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
485*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Error opening partition device");
486*9663SMark.Logan@Sun.COM 		goto error_exit;
487*9663SMark.Logan@Sun.COM 	}
488*9663SMark.Logan@Sun.COM 	/* Attach the device to the volume. */
489*9663SMark.Logan@Sun.COM 	vol->u.dev = dev;
490*9663SMark.Logan@Sun.COM 	/* Now read the bootsector. */
491*9663SMark.Logan@Sun.COM 	br = ntfs_pread(dev, 0, sizeof(NTFS_BOOT_SECTOR), bs);
492*9663SMark.Logan@Sun.COM 	if (br != sizeof(NTFS_BOOT_SECTOR)) {
493*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
494*9663SMark.Logan@Sun.COM 		if (br != -1)
495*9663SMark.Logan@Sun.COM 			errno = EINVAL;
496*9663SMark.Logan@Sun.COM 		if (!br)
497*9663SMark.Logan@Sun.COM 			ntfs_log_debug("Error: partition is smaller than "
498*9663SMark.Logan@Sun.COM 					"bootsector size. Weird!\n");
499*9663SMark.Logan@Sun.COM 		else
500*9663SMark.Logan@Sun.COM 			ntfs_log_perror("Error reading bootsector");
501*9663SMark.Logan@Sun.COM 		goto error_exit;
502*9663SMark.Logan@Sun.COM 	}
503*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
504*9663SMark.Logan@Sun.COM 	if (!ntfs_boot_sector_is_ntfs(bs, !debug)) {
505*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Error: %s is not a valid NTFS partition!\n",
506*9663SMark.Logan@Sun.COM 				dev->d_name);
507*9663SMark.Logan@Sun.COM 		errno = EINVAL;
508*9663SMark.Logan@Sun.COM 		goto error_exit;
509*9663SMark.Logan@Sun.COM 	}
510*9663SMark.Logan@Sun.COM 	if (ntfs_boot_sector_parse(vol, bs) < 0) {
511*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to parse ntfs bootsector");
512*9663SMark.Logan@Sun.COM 		goto error_exit;
513*9663SMark.Logan@Sun.COM 	}
514*9663SMark.Logan@Sun.COM 	free(bs);
515*9663SMark.Logan@Sun.COM 	bs = NULL;
516*9663SMark.Logan@Sun.COM 	/* Now set the device block size to the sector size. */
517*9663SMark.Logan@Sun.COM 	if (ntfs_device_block_size_set(vol->u.dev, vol->sector_size))
518*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Failed to set the device block size to the "
519*9663SMark.Logan@Sun.COM 				"sector size.  This may affect performance "
520*9663SMark.Logan@Sun.COM 				"but should be harmless otherwise.  Error: "
521*9663SMark.Logan@Sun.COM 				"%s\n", strerror(errno));
522*9663SMark.Logan@Sun.COM 	/*
523*9663SMark.Logan@Sun.COM 	 * We now initialize the cluster allocator.
524*9663SMark.Logan@Sun.COM 	 *
525*9663SMark.Logan@Sun.COM 	 * FIXME: Move this to its own function? (AIA)
526*9663SMark.Logan@Sun.COM 	 */
527*9663SMark.Logan@Sun.COM 
528*9663SMark.Logan@Sun.COM 	// TODO: Make this tunable at mount time. (AIA)
529*9663SMark.Logan@Sun.COM 	vol->mft_zone_multiplier = 1;
530*9663SMark.Logan@Sun.COM 
531*9663SMark.Logan@Sun.COM 	/* Determine the size of the MFT zone. */
532*9663SMark.Logan@Sun.COM 	mft_zone_size = vol->nr_clusters;
533*9663SMark.Logan@Sun.COM 	switch (vol->mft_zone_multiplier) {  /* % of volume size in clusters */
534*9663SMark.Logan@Sun.COM 	case 4:
535*9663SMark.Logan@Sun.COM 		mft_zone_size >>= 1;			/* 50%   */
536*9663SMark.Logan@Sun.COM 		break;
537*9663SMark.Logan@Sun.COM 	case 3:
538*9663SMark.Logan@Sun.COM 		mft_zone_size = mft_zone_size * 3 >> 3;	/* 37.5% */
539*9663SMark.Logan@Sun.COM 		break;
540*9663SMark.Logan@Sun.COM 	case 2:
541*9663SMark.Logan@Sun.COM 		mft_zone_size >>= 2;			/* 25%   */
542*9663SMark.Logan@Sun.COM 		break;
543*9663SMark.Logan@Sun.COM 	/* case 1: */
544*9663SMark.Logan@Sun.COM 	default:
545*9663SMark.Logan@Sun.COM 		mft_zone_size >>= 3;			/* 12.5% */
546*9663SMark.Logan@Sun.COM 		break;
547*9663SMark.Logan@Sun.COM 	}
548*9663SMark.Logan@Sun.COM 
549*9663SMark.Logan@Sun.COM 	/* Setup the mft zone. */
550*9663SMark.Logan@Sun.COM 	vol->mft_zone_start = vol->mft_zone_pos = vol->mft_lcn;
551*9663SMark.Logan@Sun.COM 	ntfs_log_debug("mft_zone_pos = 0x%llx\n", (long long)vol->mft_zone_pos);
552*9663SMark.Logan@Sun.COM 
553*9663SMark.Logan@Sun.COM 	/*
554*9663SMark.Logan@Sun.COM 	 * Calculate the mft_lcn for an unmodified NTFS volume (see mkntfs
555*9663SMark.Logan@Sun.COM 	 * source) and if the actual mft_lcn is in the expected place or even
556*9663SMark.Logan@Sun.COM 	 * further to the front of the volume, extend the mft_zone to cover the
557*9663SMark.Logan@Sun.COM 	 * beginning of the volume as well. This is in order to protect the
558*9663SMark.Logan@Sun.COM 	 * area reserved for the mft bitmap as well within the mft_zone itself.
559*9663SMark.Logan@Sun.COM 	 * On non-standard volumes we don't protect it as the overhead would be
560*9663SMark.Logan@Sun.COM 	 * higher than the speed increase we would get by doing it.
561*9663SMark.Logan@Sun.COM 	 */
562*9663SMark.Logan@Sun.COM 	mft_lcn = (8192 + 2 * vol->cluster_size - 1) / vol->cluster_size;
563*9663SMark.Logan@Sun.COM 	if (mft_lcn * vol->cluster_size < 16 * 1024)
564*9663SMark.Logan@Sun.COM 		mft_lcn = (16 * 1024 + vol->cluster_size - 1) /
565*9663SMark.Logan@Sun.COM 				vol->cluster_size;
566*9663SMark.Logan@Sun.COM 	if (vol->mft_zone_start <= mft_lcn)
567*9663SMark.Logan@Sun.COM 		vol->mft_zone_start = 0;
568*9663SMark.Logan@Sun.COM 	ntfs_log_debug("mft_zone_start = 0x%llx\n",
569*9663SMark.Logan@Sun.COM 			(long long)vol->mft_zone_start);
570*9663SMark.Logan@Sun.COM 
571*9663SMark.Logan@Sun.COM 	/*
572*9663SMark.Logan@Sun.COM 	 * Need to cap the mft zone on non-standard volumes so that it does
573*9663SMark.Logan@Sun.COM 	 * not point outside the boundaries of the volume. We do this by
574*9663SMark.Logan@Sun.COM 	 * halving the zone size until we are inside the volume.
575*9663SMark.Logan@Sun.COM 	 */
576*9663SMark.Logan@Sun.COM 	vol->mft_zone_end = vol->mft_lcn + mft_zone_size;
577*9663SMark.Logan@Sun.COM 	while (vol->mft_zone_end >= vol->nr_clusters) {
578*9663SMark.Logan@Sun.COM 		mft_zone_size >>= 1;
579*9663SMark.Logan@Sun.COM 		vol->mft_zone_end = vol->mft_lcn + mft_zone_size;
580*9663SMark.Logan@Sun.COM 	}
581*9663SMark.Logan@Sun.COM 	ntfs_log_debug("mft_zone_end = 0x%llx\n", (long long)vol->mft_zone_end);
582*9663SMark.Logan@Sun.COM 
583*9663SMark.Logan@Sun.COM 	/*
584*9663SMark.Logan@Sun.COM 	 * Set the current position within each data zone to the start of the
585*9663SMark.Logan@Sun.COM 	 * respective zone.
586*9663SMark.Logan@Sun.COM 	 */
587*9663SMark.Logan@Sun.COM 	vol->data1_zone_pos = vol->mft_zone_end;
588*9663SMark.Logan@Sun.COM 	ntfs_log_debug("data1_zone_pos = 0x%llx\n", vol->data1_zone_pos);
589*9663SMark.Logan@Sun.COM 	vol->data2_zone_pos = 0;
590*9663SMark.Logan@Sun.COM 	ntfs_log_debug("data2_zone_pos = 0x%llx\n", vol->data2_zone_pos);
591*9663SMark.Logan@Sun.COM 
592*9663SMark.Logan@Sun.COM 	/* Set the mft data allocation position to mft record 24. */
593*9663SMark.Logan@Sun.COM 	vol->mft_data_pos = 24;
594*9663SMark.Logan@Sun.COM 
595*9663SMark.Logan@Sun.COM 	/*
596*9663SMark.Logan@Sun.COM 	 * The cluster allocator is now fully operational.
597*9663SMark.Logan@Sun.COM 	 */
598*9663SMark.Logan@Sun.COM 
599*9663SMark.Logan@Sun.COM 	/* Need to setup $MFT so we can use the library read functions. */
600*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Loading $MFT... ");
601*9663SMark.Logan@Sun.COM 	if (ntfs_mft_load(vol) < 0) {
602*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
603*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to load $MFT");
604*9663SMark.Logan@Sun.COM 		goto error_exit;
605*9663SMark.Logan@Sun.COM 	}
606*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
607*9663SMark.Logan@Sun.COM 
608*9663SMark.Logan@Sun.COM 	/* Need to setup $MFTMirr so we can use the write functions, too. */
609*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Loading $MFTMirr... ");
610*9663SMark.Logan@Sun.COM 	if (ntfs_mftmirr_load(vol) < 0) {
611*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
612*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to load $MFTMirr");
613*9663SMark.Logan@Sun.COM 		goto error_exit;
614*9663SMark.Logan@Sun.COM 	}
615*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
616*9663SMark.Logan@Sun.COM 	return vol;
617*9663SMark.Logan@Sun.COM error_exit:
618*9663SMark.Logan@Sun.COM 	eo = errno;
619*9663SMark.Logan@Sun.COM 	free(bs);
620*9663SMark.Logan@Sun.COM 	if (vol)
621*9663SMark.Logan@Sun.COM 		__ntfs_volume_release(vol);
622*9663SMark.Logan@Sun.COM 	errno = eo;
623*9663SMark.Logan@Sun.COM 	return NULL;
624*9663SMark.Logan@Sun.COM }
625*9663SMark.Logan@Sun.COM 
626*9663SMark.Logan@Sun.COM /**
627*9663SMark.Logan@Sun.COM  * ntfs_volume_check_logfile - check logfile on target volume
628*9663SMark.Logan@Sun.COM  * @vol:	volume on which to check logfile
629*9663SMark.Logan@Sun.COM  *
630*9663SMark.Logan@Sun.COM  * Return 0 on success and -1 on error with errno set error code.
631*9663SMark.Logan@Sun.COM  */
632*9663SMark.Logan@Sun.COM static int ntfs_volume_check_logfile(ntfs_volume *vol)
633*9663SMark.Logan@Sun.COM {
634*9663SMark.Logan@Sun.COM 	ntfs_inode *ni;
635*9663SMark.Logan@Sun.COM 	ntfs_attr *na = NULL;
636*9663SMark.Logan@Sun.COM 	RESTART_PAGE_HEADER *rp = NULL;
637*9663SMark.Logan@Sun.COM 	int err = 0;
638*9663SMark.Logan@Sun.COM 
639*9663SMark.Logan@Sun.COM 	if ((ni = ntfs_inode_open(vol, FILE_LogFile)) == NULL) {
640*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Failed to open inode FILE_LogFile.\n");
641*9663SMark.Logan@Sun.COM 		errno = EIO;
642*9663SMark.Logan@Sun.COM 		return -1;
643*9663SMark.Logan@Sun.COM 	}
644*9663SMark.Logan@Sun.COM 	if ((na = ntfs_attr_open(ni, AT_DATA, AT_UNNAMED, 0)) == NULL) {
645*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Failed to open $FILE_LogFile/$DATA\n");
646*9663SMark.Logan@Sun.COM 		err = EIO;
647*9663SMark.Logan@Sun.COM 		goto exit;
648*9663SMark.Logan@Sun.COM 	}
649*9663SMark.Logan@Sun.COM 	if (!ntfs_check_logfile(na, &rp) || !ntfs_is_logfile_clean(na, rp))
650*9663SMark.Logan@Sun.COM 		err = EOPNOTSUPP;
651*9663SMark.Logan@Sun.COM 	free(rp);
652*9663SMark.Logan@Sun.COM exit:
653*9663SMark.Logan@Sun.COM 	if (na)
654*9663SMark.Logan@Sun.COM 		ntfs_attr_close(na);
655*9663SMark.Logan@Sun.COM 	ntfs_inode_close(ni);
656*9663SMark.Logan@Sun.COM 	if (err) {
657*9663SMark.Logan@Sun.COM 		errno = err;
658*9663SMark.Logan@Sun.COM 		return -1;
659*9663SMark.Logan@Sun.COM 	}
660*9663SMark.Logan@Sun.COM 	return 0;
661*9663SMark.Logan@Sun.COM }
662*9663SMark.Logan@Sun.COM 
663*9663SMark.Logan@Sun.COM /**
664*9663SMark.Logan@Sun.COM  * ntfs_hiberfile_open - Find and open '/hiberfil.sys'
665*9663SMark.Logan@Sun.COM  * @vol:    An ntfs volume obtained from ntfs_mount
666*9663SMark.Logan@Sun.COM  *
667*9663SMark.Logan@Sun.COM  * Return:  inode  Success, hiberfil.sys is valid
668*9663SMark.Logan@Sun.COM  *	    NULL   hiberfil.sys doesn't exist or some other error occurred
669*9663SMark.Logan@Sun.COM  */
670*9663SMark.Logan@Sun.COM static ntfs_inode *ntfs_hiberfile_open(ntfs_volume *vol)
671*9663SMark.Logan@Sun.COM {
672*9663SMark.Logan@Sun.COM 	u64 inode;
673*9663SMark.Logan@Sun.COM 	ntfs_inode *ni_root;
674*9663SMark.Logan@Sun.COM 	ntfs_inode *ni_hibr = NULL;
675*9663SMark.Logan@Sun.COM 	ntfschar   *unicode = NULL;
676*9663SMark.Logan@Sun.COM 	int unicode_len;
677*9663SMark.Logan@Sun.COM 	const char *hiberfile = "hiberfil.sys";
678*9663SMark.Logan@Sun.COM 
679*9663SMark.Logan@Sun.COM 	if (!vol) {
680*9663SMark.Logan@Sun.COM 		errno = EINVAL;
681*9663SMark.Logan@Sun.COM 		return NULL;
682*9663SMark.Logan@Sun.COM 	}
683*9663SMark.Logan@Sun.COM 
684*9663SMark.Logan@Sun.COM 	ni_root = ntfs_inode_open(vol, FILE_root);
685*9663SMark.Logan@Sun.COM 	if (!ni_root) {
686*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Couldn't open the root directory.\n");
687*9663SMark.Logan@Sun.COM 		return NULL;
688*9663SMark.Logan@Sun.COM 	}
689*9663SMark.Logan@Sun.COM 
690*9663SMark.Logan@Sun.COM 	unicode_len = ntfs_mbstoucs(hiberfile, &unicode, 0);
691*9663SMark.Logan@Sun.COM 	if (unicode_len < 0) {
692*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Couldn't convert 'hiberfil.sys' to Unicode");
693*9663SMark.Logan@Sun.COM 		goto out;
694*9663SMark.Logan@Sun.COM 	}
695*9663SMark.Logan@Sun.COM 
696*9663SMark.Logan@Sun.COM 	inode = ntfs_inode_lookup_by_name(ni_root, unicode, unicode_len);
697*9663SMark.Logan@Sun.COM 	if (inode == (u64)-1) {
698*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Couldn't find file '%s'.\n", hiberfile);
699*9663SMark.Logan@Sun.COM 		goto out;
700*9663SMark.Logan@Sun.COM 	}
701*9663SMark.Logan@Sun.COM 
702*9663SMark.Logan@Sun.COM 	inode = MREF(inode);
703*9663SMark.Logan@Sun.COM 	ni_hibr = ntfs_inode_open(vol, inode);
704*9663SMark.Logan@Sun.COM 	if (!ni_hibr) {
705*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Couldn't open inode %lld.\n", (long long)inode);
706*9663SMark.Logan@Sun.COM 		goto out;
707*9663SMark.Logan@Sun.COM 	}
708*9663SMark.Logan@Sun.COM out:
709*9663SMark.Logan@Sun.COM 	ntfs_inode_close(ni_root);
710*9663SMark.Logan@Sun.COM 	free(unicode);
711*9663SMark.Logan@Sun.COM 	return ni_hibr;
712*9663SMark.Logan@Sun.COM }
713*9663SMark.Logan@Sun.COM 
714*9663SMark.Logan@Sun.COM 
715*9663SMark.Logan@Sun.COM #define NTFS_HIBERFILE_HEADER_SIZE	4096
716*9663SMark.Logan@Sun.COM 
717*9663SMark.Logan@Sun.COM /**
718*9663SMark.Logan@Sun.COM  * ntfs_volume_check_hiberfile - check hiberfil.sys whether Windows is
719*9663SMark.Logan@Sun.COM  *                               hibernated on the target volume
720*9663SMark.Logan@Sun.COM  * @vol:    volume on which to check hiberfil.sys
721*9663SMark.Logan@Sun.COM  *
722*9663SMark.Logan@Sun.COM  * Return:  0 if Windows isn't hibernated for sure
723*9663SMark.Logan@Sun.COM  *         -1 otherwise and errno is set to the appropriate value
724*9663SMark.Logan@Sun.COM  */
725*9663SMark.Logan@Sun.COM static int ntfs_volume_check_hiberfile(ntfs_volume *vol)
726*9663SMark.Logan@Sun.COM {
727*9663SMark.Logan@Sun.COM 	ntfs_inode *ni;
728*9663SMark.Logan@Sun.COM 	ntfs_attr *na = NULL;
729*9663SMark.Logan@Sun.COM 	int bytes_read, ret = -1;
730*9663SMark.Logan@Sun.COM 	char *buf = NULL;
731*9663SMark.Logan@Sun.COM 
732*9663SMark.Logan@Sun.COM 	ni = ntfs_hiberfile_open(vol);
733*9663SMark.Logan@Sun.COM 	if (!ni) {
734*9663SMark.Logan@Sun.COM 		if (errno == ENOENT)
735*9663SMark.Logan@Sun.COM 			return 0;
736*9663SMark.Logan@Sun.COM 		return -1;
737*9663SMark.Logan@Sun.COM 	}
738*9663SMark.Logan@Sun.COM 
739*9663SMark.Logan@Sun.COM 	buf = ntfs_malloc(NTFS_HIBERFILE_HEADER_SIZE);
740*9663SMark.Logan@Sun.COM 	if (!buf)
741*9663SMark.Logan@Sun.COM 		goto out;
742*9663SMark.Logan@Sun.COM 
743*9663SMark.Logan@Sun.COM 	na = ntfs_attr_open(ni, AT_DATA, AT_UNNAMED, 0);
744*9663SMark.Logan@Sun.COM 	if (!na) {
745*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open hiberfil.sys data attribute");
746*9663SMark.Logan@Sun.COM 		goto out;
747*9663SMark.Logan@Sun.COM 	}
748*9663SMark.Logan@Sun.COM 
749*9663SMark.Logan@Sun.COM 	bytes_read = ntfs_attr_pread(na, 0, NTFS_HIBERFILE_HEADER_SIZE, buf);
750*9663SMark.Logan@Sun.COM 	if (bytes_read == -1) {
751*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to read hiberfil.sys");
752*9663SMark.Logan@Sun.COM 		goto out;
753*9663SMark.Logan@Sun.COM 	}
754*9663SMark.Logan@Sun.COM 	if (bytes_read < NTFS_HIBERFILE_HEADER_SIZE) {
755*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Hibernated non-system partition, refused to "
756*9663SMark.Logan@Sun.COM 				"mount!\n");
757*9663SMark.Logan@Sun.COM 		errno = EPERM;
758*9663SMark.Logan@Sun.COM 		goto out;
759*9663SMark.Logan@Sun.COM 	}
760*9663SMark.Logan@Sun.COM 	if (memcmp(buf, "hibr", 4) == 0) {
761*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Windows is hibernated, refused to mount!\n");
762*9663SMark.Logan@Sun.COM 		errno = EPERM;
763*9663SMark.Logan@Sun.COM 		goto out;
764*9663SMark.Logan@Sun.COM 	}
765*9663SMark.Logan@Sun.COM 	ret = 0;
766*9663SMark.Logan@Sun.COM out:
767*9663SMark.Logan@Sun.COM 	if (na)
768*9663SMark.Logan@Sun.COM 		ntfs_attr_close(na);
769*9663SMark.Logan@Sun.COM 	free(buf);
770*9663SMark.Logan@Sun.COM 	ntfs_inode_close(ni);
771*9663SMark.Logan@Sun.COM 	return ret;
772*9663SMark.Logan@Sun.COM }
773*9663SMark.Logan@Sun.COM 
774*9663SMark.Logan@Sun.COM /**
775*9663SMark.Logan@Sun.COM  * ntfs_volume_get_nr_free_mft_records - calculate number of free MFT records
776*9663SMark.Logan@Sun.COM  * vol:		ntfs volume for which perform calculations.
777*9663SMark.Logan@Sun.COM  *
778*9663SMark.Logan@Sun.COM  * This function initializes @vol->nr_free_mft_records. @vol->mftbmp_na should
779*9663SMark.Logan@Sun.COM  * be already opened upon call to this function.
780*9663SMark.Logan@Sun.COM  *
781*9663SMark.Logan@Sun.COM  * Return 0 on success. On error return -1 with errno set appropriately and
782*9663SMark.Logan@Sun.COM  * @vol->nr_free_mft_records is not touched in this case.
783*9663SMark.Logan@Sun.COM  */
784*9663SMark.Logan@Sun.COM static int ntfs_volume_get_nr_free_mft_records(ntfs_volume *vol)
785*9663SMark.Logan@Sun.COM {
786*9663SMark.Logan@Sun.COM 	long nr_free = vol->mft_na->data_size >> vol->mft_record_size_bits;
787*9663SMark.Logan@Sun.COM 	s64 br, total = 0;
788*9663SMark.Logan@Sun.COM 	u8 *buf;
789*9663SMark.Logan@Sun.COM 
790*9663SMark.Logan@Sun.COM 	buf = ntfs_malloc(vol->cluster_size);
791*9663SMark.Logan@Sun.COM 	if (!buf)
792*9663SMark.Logan@Sun.COM 		return -1;
793*9663SMark.Logan@Sun.COM 	while (1) {
794*9663SMark.Logan@Sun.COM 		int i, j;
795*9663SMark.Logan@Sun.COM 
796*9663SMark.Logan@Sun.COM 		br = ntfs_attr_pread(vol->mftbmp_na, total,
797*9663SMark.Logan@Sun.COM 				vol->cluster_size, buf);
798*9663SMark.Logan@Sun.COM 		if (br <= 0)
799*9663SMark.Logan@Sun.COM 			break;
800*9663SMark.Logan@Sun.COM 		total += br;
801*9663SMark.Logan@Sun.COM 		for (i = 0; i < br; i++)
802*9663SMark.Logan@Sun.COM 			for (j = 0; j < 8; j++)
803*9663SMark.Logan@Sun.COM 				if ((buf[i] >> j) & 1)
804*9663SMark.Logan@Sun.COM 					nr_free--;
805*9663SMark.Logan@Sun.COM 	}
806*9663SMark.Logan@Sun.COM 	free(buf);
807*9663SMark.Logan@Sun.COM 	if (!total || br < 0) {
808*9663SMark.Logan@Sun.COM 		ntfs_log_error("pread: %s\n", strerror(errno));
809*9663SMark.Logan@Sun.COM 		return -1;
810*9663SMark.Logan@Sun.COM 	}
811*9663SMark.Logan@Sun.COM 	vol->nr_free_mft_records = nr_free;
812*9663SMark.Logan@Sun.COM 	return 0;
813*9663SMark.Logan@Sun.COM }
814*9663SMark.Logan@Sun.COM 
815*9663SMark.Logan@Sun.COM /**
816*9663SMark.Logan@Sun.COM  * ntfs_volume_get_nr_free_clusters - calculate number of free clusters
817*9663SMark.Logan@Sun.COM  * vol:		ntfs volume for which perform calculations.
818*9663SMark.Logan@Sun.COM  *
819*9663SMark.Logan@Sun.COM  * This function initializes @vol->nr_free_clusters. @vol->lcnbmp_na should be
820*9663SMark.Logan@Sun.COM  * already opened upon call to this function.
821*9663SMark.Logan@Sun.COM  *
822*9663SMark.Logan@Sun.COM  * Return 0 on success. On error return -1 with errno set appropriately and
823*9663SMark.Logan@Sun.COM  * @vol->nr_free_clusters is not touched in this case.
824*9663SMark.Logan@Sun.COM  */
825*9663SMark.Logan@Sun.COM static long ntfs_volume_get_nr_free_clusters(ntfs_volume *vol)
826*9663SMark.Logan@Sun.COM {
827*9663SMark.Logan@Sun.COM 	long nr_free = vol->nr_clusters;
828*9663SMark.Logan@Sun.COM 	s64 br, total = 0;
829*9663SMark.Logan@Sun.COM 	u8 *buf;
830*9663SMark.Logan@Sun.COM 
831*9663SMark.Logan@Sun.COM 	buf = ntfs_malloc(vol->cluster_size);
832*9663SMark.Logan@Sun.COM 	if (!buf)
833*9663SMark.Logan@Sun.COM 		return -1;
834*9663SMark.Logan@Sun.COM 	while (1) {
835*9663SMark.Logan@Sun.COM 		int i, j;
836*9663SMark.Logan@Sun.COM 
837*9663SMark.Logan@Sun.COM 		br = ntfs_attr_pread(vol->lcnbmp_na, total,
838*9663SMark.Logan@Sun.COM 				vol->cluster_size, buf);
839*9663SMark.Logan@Sun.COM 		if (br <= 0)
840*9663SMark.Logan@Sun.COM 			break;
841*9663SMark.Logan@Sun.COM 		total += br;
842*9663SMark.Logan@Sun.COM 		for (i = 0; i < br; i++)
843*9663SMark.Logan@Sun.COM 			for (j = 0; j < 8; j++)
844*9663SMark.Logan@Sun.COM 				if ((buf[i] >> j) & 1)
845*9663SMark.Logan@Sun.COM 					nr_free--;
846*9663SMark.Logan@Sun.COM 	}
847*9663SMark.Logan@Sun.COM 	free(buf);
848*9663SMark.Logan@Sun.COM 	if (!total || br < 0) {
849*9663SMark.Logan@Sun.COM 		ntfs_log_error("pread: %s\n", strerror(errno));
850*9663SMark.Logan@Sun.COM 		return -1;
851*9663SMark.Logan@Sun.COM 	}
852*9663SMark.Logan@Sun.COM 	vol->nr_free_clusters = nr_free;
853*9663SMark.Logan@Sun.COM 	return 0;
854*9663SMark.Logan@Sun.COM }
855*9663SMark.Logan@Sun.COM 
856*9663SMark.Logan@Sun.COM /**
857*9663SMark.Logan@Sun.COM  * ntfs_device_mount - open ntfs volume
858*9663SMark.Logan@Sun.COM  * @dev:	device to open
859*9663SMark.Logan@Sun.COM  * @flags:	optional mount flags
860*9663SMark.Logan@Sun.COM  *
861*9663SMark.Logan@Sun.COM  * This function mounts an ntfs volume. @dev should describe the device which
862*9663SMark.Logan@Sun.COM  * to mount as the ntfs volume.
863*9663SMark.Logan@Sun.COM  *
864*9663SMark.Logan@Sun.COM  * @flags is an optional second parameter. Some flags are similar to flags used
865*9663SMark.Logan@Sun.COM  * as for the mount system call (man 2 mount). Currently the following flags
866*9663SMark.Logan@Sun.COM  * are implemented:
867*9663SMark.Logan@Sun.COM  *	NTFS_MNT_RDONLY		- mount volume read-only
868*9663SMark.Logan@Sun.COM  *	NTFS_MNT_CASE_SENSITIVE - treat filenames as case sensitive even if
869*9663SMark.Logan@Sun.COM  *				  they are not in POSIX namespace
870*9663SMark.Logan@Sun.COM  *	NTFS_MNT_NOT_EXCLUSIVE	- (unix only) do not open volume exclusively
871*9663SMark.Logan@Sun.COM  *	NTFS_MNT_FORENSIC	- mount for forensic purposes, i.e. do not do
872*9663SMark.Logan@Sun.COM  *				  any writing at all during the mount, i.e. no
873*9663SMark.Logan@Sun.COM  *				  journal emptying, no dirty bit setting, etc.
874*9663SMark.Logan@Sun.COM  *	NTFS_MNT_INTERIX	- make libntfs recognize special Interix files
875*9663SMark.Logan@Sun.COM  *
876*9663SMark.Logan@Sun.COM  * The function opens the device @dev and verifies that it contains a valid
877*9663SMark.Logan@Sun.COM  * bootsector. Then, it allocates an ntfs_volume structure and initializes
878*9663SMark.Logan@Sun.COM  * some of the values inside the structure from the information stored in the
879*9663SMark.Logan@Sun.COM  * bootsector. It proceeds to load the necessary system files and completes
880*9663SMark.Logan@Sun.COM  * setting up the structure.
881*9663SMark.Logan@Sun.COM  *
882*9663SMark.Logan@Sun.COM  * Return the allocated volume structure on success and NULL on error with
883*9663SMark.Logan@Sun.COM  * errno set to the error code.
884*9663SMark.Logan@Sun.COM  */
885*9663SMark.Logan@Sun.COM ntfs_volume *ntfs_device_mount(struct ntfs_device *dev, ntfs_mount_flags flags)
886*9663SMark.Logan@Sun.COM {
887*9663SMark.Logan@Sun.COM 	s64 l;
888*9663SMark.Logan@Sun.COM #ifdef DEBUG
889*9663SMark.Logan@Sun.COM 	const char *OK = "OK\n";
890*9663SMark.Logan@Sun.COM 	const char *FAILED = "FAILED\n";
891*9663SMark.Logan@Sun.COM #endif
892*9663SMark.Logan@Sun.COM 	ntfs_volume *vol;
893*9663SMark.Logan@Sun.COM 	u8 *m = NULL, *m2 = NULL;
894*9663SMark.Logan@Sun.COM 	ntfs_attr_search_ctx *ctx = NULL;
895*9663SMark.Logan@Sun.COM 	ntfs_inode *ni;
896*9663SMark.Logan@Sun.COM 	ntfs_attr *na;
897*9663SMark.Logan@Sun.COM 	ATTR_RECORD *a;
898*9663SMark.Logan@Sun.COM 	VOLUME_INFORMATION *vinf;
899*9663SMark.Logan@Sun.COM 	ntfschar *vname;
900*9663SMark.Logan@Sun.COM 	int i, j, eo;
901*9663SMark.Logan@Sun.COM 	u32 u;
902*9663SMark.Logan@Sun.COM 
903*9663SMark.Logan@Sun.COM 	vol = ntfs_volume_startup(dev, flags);
904*9663SMark.Logan@Sun.COM 	if (!vol) {
905*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to startup volume");
906*9663SMark.Logan@Sun.COM 		return NULL;
907*9663SMark.Logan@Sun.COM 	}
908*9663SMark.Logan@Sun.COM 	/* Record whether this is a forensic mount. */
909*9663SMark.Logan@Sun.COM 	if (flags & NTFS_MNT_FORENSIC)
910*9663SMark.Logan@Sun.COM 		NVolSetForensicMount(vol);
911*9663SMark.Logan@Sun.COM 	/* Load data from $MFT and $MFTMirr and compare the contents. */
912*9663SMark.Logan@Sun.COM 	m  = (u8*)ntfs_malloc(vol->mftmirr_size << vol->mft_record_size_bits);
913*9663SMark.Logan@Sun.COM 	m2 = (u8*)ntfs_malloc(vol->mftmirr_size << vol->mft_record_size_bits);
914*9663SMark.Logan@Sun.COM 	if (!m || !m2)
915*9663SMark.Logan@Sun.COM 		goto error_exit;
916*9663SMark.Logan@Sun.COM 
917*9663SMark.Logan@Sun.COM 	l = ntfs_attr_mst_pread(vol->mft_na, 0, vol->mftmirr_size,
918*9663SMark.Logan@Sun.COM 			vol->mft_record_size, m);
919*9663SMark.Logan@Sun.COM 	if (l != vol->mftmirr_size) {
920*9663SMark.Logan@Sun.COM 		if (l == -1)
921*9663SMark.Logan@Sun.COM 			ntfs_log_perror("Failed to read $MFT");
922*9663SMark.Logan@Sun.COM 		else {
923*9663SMark.Logan@Sun.COM 			ntfs_log_debug("Failed to read $MFT, unexpected length "
924*9663SMark.Logan@Sun.COM 				       "(%d != %lld).\n", vol->mftmirr_size, l);
925*9663SMark.Logan@Sun.COM 			errno = EIO;
926*9663SMark.Logan@Sun.COM 		}
927*9663SMark.Logan@Sun.COM 		goto error_exit;
928*9663SMark.Logan@Sun.COM 	}
929*9663SMark.Logan@Sun.COM 	l = ntfs_attr_mst_pread(vol->mftmirr_na, 0, vol->mftmirr_size,
930*9663SMark.Logan@Sun.COM 			vol->mft_record_size, m2);
931*9663SMark.Logan@Sun.COM 	if (l != vol->mftmirr_size) {
932*9663SMark.Logan@Sun.COM 		if (l == -1)
933*9663SMark.Logan@Sun.COM 			ntfs_log_perror("Failed to read $MFTMirr");
934*9663SMark.Logan@Sun.COM 		else {
935*9663SMark.Logan@Sun.COM 			ntfs_log_debug("Failed to read $MFTMirr, unexpected "
936*9663SMark.Logan@Sun.COM 				       "length (%d != %lld).\n",
937*9663SMark.Logan@Sun.COM 				       vol->mftmirr_size, l);
938*9663SMark.Logan@Sun.COM 			errno = EIO;
939*9663SMark.Logan@Sun.COM 		}
940*9663SMark.Logan@Sun.COM 		goto error_exit;
941*9663SMark.Logan@Sun.COM 	}
942*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Comparing $MFTMirr to $MFT... ");
943*9663SMark.Logan@Sun.COM 	for (i = 0; i < vol->mftmirr_size; ++i) {
944*9663SMark.Logan@Sun.COM 		MFT_RECORD *mrec, *mrec2;
945*9663SMark.Logan@Sun.COM 		const char *ESTR[12] = { "$MFT", "$MFTMirr", "$LogFile",
946*9663SMark.Logan@Sun.COM 			"$Volume", "$AttrDef", "root directory", "$Bitmap",
947*9663SMark.Logan@Sun.COM 			"$Boot", "$BadClus", "$Secure", "$UpCase", "$Extend" };
948*9663SMark.Logan@Sun.COM 		const char *s;
949*9663SMark.Logan@Sun.COM 
950*9663SMark.Logan@Sun.COM 		if (i < 12)
951*9663SMark.Logan@Sun.COM 			s = ESTR[i];
952*9663SMark.Logan@Sun.COM 		else if (i < 16)
953*9663SMark.Logan@Sun.COM 			s = "system file";
954*9663SMark.Logan@Sun.COM 		else
955*9663SMark.Logan@Sun.COM 			s = "mft record";
956*9663SMark.Logan@Sun.COM 
957*9663SMark.Logan@Sun.COM 		mrec = (MFT_RECORD*)(m + i * vol->mft_record_size);
958*9663SMark.Logan@Sun.COM 		if (mrec->flags & MFT_RECORD_IN_USE) {
959*9663SMark.Logan@Sun.COM 			if (ntfs_is_baad_record(mrec->magic)) {
960*9663SMark.Logan@Sun.COM 				ntfs_log_debug("FAILED\n");
961*9663SMark.Logan@Sun.COM 				ntfs_log_debug("$MFT error: Incomplete multi "
962*9663SMark.Logan@Sun.COM 						"sector transfer detected in "
963*9663SMark.Logan@Sun.COM 						"%s.\n", s);
964*9663SMark.Logan@Sun.COM 				goto io_error_exit;
965*9663SMark.Logan@Sun.COM 			}
966*9663SMark.Logan@Sun.COM 			if (!ntfs_is_mft_record(mrec->magic)) {
967*9663SMark.Logan@Sun.COM 				ntfs_log_debug("FAILED\n");
968*9663SMark.Logan@Sun.COM 				ntfs_log_debug("$MFT error: Invalid mft "
969*9663SMark.Logan@Sun.COM 						"record for %s.\n", s);
970*9663SMark.Logan@Sun.COM 				goto io_error_exit;
971*9663SMark.Logan@Sun.COM 			}
972*9663SMark.Logan@Sun.COM 		}
973*9663SMark.Logan@Sun.COM 		mrec2 = (MFT_RECORD*)(m2 + i * vol->mft_record_size);
974*9663SMark.Logan@Sun.COM 		if (mrec2->flags & MFT_RECORD_IN_USE) {
975*9663SMark.Logan@Sun.COM 			if (ntfs_is_baad_record(mrec2->magic)) {
976*9663SMark.Logan@Sun.COM 				ntfs_log_debug("FAILED\n");
977*9663SMark.Logan@Sun.COM 				ntfs_log_debug("$MFTMirr error: Incomplete "
978*9663SMark.Logan@Sun.COM 						"multi sector transfer "
979*9663SMark.Logan@Sun.COM 						"detected in %s.\n", s);
980*9663SMark.Logan@Sun.COM 				goto io_error_exit;
981*9663SMark.Logan@Sun.COM 			}
982*9663SMark.Logan@Sun.COM 			if (!ntfs_is_mft_record(mrec2->magic)) {
983*9663SMark.Logan@Sun.COM 				ntfs_log_debug("FAILED\n");
984*9663SMark.Logan@Sun.COM 				ntfs_log_debug("$MFTMirr error: Invalid mft "
985*9663SMark.Logan@Sun.COM 						"record for %s.\n", s);
986*9663SMark.Logan@Sun.COM 				goto io_error_exit;
987*9663SMark.Logan@Sun.COM 			}
988*9663SMark.Logan@Sun.COM 		}
989*9663SMark.Logan@Sun.COM 		if (memcmp(mrec, mrec2, ntfs_mft_record_get_data_size(mrec))) {
990*9663SMark.Logan@Sun.COM 			ntfs_log_debug(FAILED);
991*9663SMark.Logan@Sun.COM 			ntfs_log_debug("$MFTMirr does not match $MFT. Run "
992*9663SMark.Logan@Sun.COM 					"chkdsk.\n");
993*9663SMark.Logan@Sun.COM 			goto io_error_exit;
994*9663SMark.Logan@Sun.COM 		}
995*9663SMark.Logan@Sun.COM 	}
996*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
997*9663SMark.Logan@Sun.COM 
998*9663SMark.Logan@Sun.COM 	free(m2);
999*9663SMark.Logan@Sun.COM 	free(m);
1000*9663SMark.Logan@Sun.COM 	m = m2 = NULL;
1001*9663SMark.Logan@Sun.COM 
1002*9663SMark.Logan@Sun.COM 	/* Now load the bitmap from $Bitmap. */
1003*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Loading $Bitmap... ");
1004*9663SMark.Logan@Sun.COM 	vol->lcnbmp_ni = ntfs_inode_open(vol, FILE_Bitmap);
1005*9663SMark.Logan@Sun.COM 	if (!vol->lcnbmp_ni) {
1006*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1007*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open inode");
1008*9663SMark.Logan@Sun.COM 		goto error_exit;
1009*9663SMark.Logan@Sun.COM 	}
1010*9663SMark.Logan@Sun.COM 	/* Get an ntfs attribute for $Bitmap/$DATA. */
1011*9663SMark.Logan@Sun.COM 	vol->lcnbmp_na = ntfs_attr_open(vol->lcnbmp_ni, AT_DATA, AT_UNNAMED, 0);
1012*9663SMark.Logan@Sun.COM 	if (!vol->lcnbmp_na) {
1013*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1014*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open ntfs attribute");
1015*9663SMark.Logan@Sun.COM 		goto error_exit;
1016*9663SMark.Logan@Sun.COM 	}
1017*9663SMark.Logan@Sun.COM 	/* Done with the $Bitmap mft record. */
1018*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
1019*9663SMark.Logan@Sun.COM 
1020*9663SMark.Logan@Sun.COM 	/* Now load the upcase table from $UpCase. */
1021*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Loading $UpCase... ");
1022*9663SMark.Logan@Sun.COM 	ni = ntfs_inode_open(vol, FILE_UpCase);
1023*9663SMark.Logan@Sun.COM 	if (!ni) {
1024*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1025*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open inode");
1026*9663SMark.Logan@Sun.COM 		goto error_exit;
1027*9663SMark.Logan@Sun.COM 	}
1028*9663SMark.Logan@Sun.COM 	/* Get an ntfs attribute for $UpCase/$DATA. */
1029*9663SMark.Logan@Sun.COM 	na = ntfs_attr_open(ni, AT_DATA, AT_UNNAMED, 0);
1030*9663SMark.Logan@Sun.COM 	if (!na) {
1031*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1032*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open ntfs attribute");
1033*9663SMark.Logan@Sun.COM 		goto error_exit;
1034*9663SMark.Logan@Sun.COM 	}
1035*9663SMark.Logan@Sun.COM 	/*
1036*9663SMark.Logan@Sun.COM 	 * Note: Normally, the upcase table has a length equal to 65536
1037*9663SMark.Logan@Sun.COM 	 * 2-byte Unicode characters but allow for different cases, so no
1038*9663SMark.Logan@Sun.COM 	 * checks done. Just check we don't overflow 32-bits worth of Unicode
1039*9663SMark.Logan@Sun.COM 	 * characters.
1040*9663SMark.Logan@Sun.COM 	 */
1041*9663SMark.Logan@Sun.COM 	if (na->data_size & ~0x1ffffffffULL) {
1042*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1043*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Error: Upcase table is too big (max 32-bit "
1044*9663SMark.Logan@Sun.COM 				"allowed).\n");
1045*9663SMark.Logan@Sun.COM 		errno = EINVAL;
1046*9663SMark.Logan@Sun.COM 		goto error_exit;
1047*9663SMark.Logan@Sun.COM 	}
1048*9663SMark.Logan@Sun.COM 	if (vol->upcase_len != na->data_size >> 1) {
1049*9663SMark.Logan@Sun.COM 		vol->upcase_len = na->data_size >> 1;
1050*9663SMark.Logan@Sun.COM 		/* Throw away default table. */
1051*9663SMark.Logan@Sun.COM 		free(vol->upcase);
1052*9663SMark.Logan@Sun.COM 		vol->upcase = (ntfschar*)ntfs_malloc(na->data_size);
1053*9663SMark.Logan@Sun.COM 		if (!vol->upcase) {
1054*9663SMark.Logan@Sun.COM 			ntfs_log_debug(FAILED);
1055*9663SMark.Logan@Sun.COM 			goto error_exit;
1056*9663SMark.Logan@Sun.COM 		}
1057*9663SMark.Logan@Sun.COM 	}
1058*9663SMark.Logan@Sun.COM 	/* Read in the $DATA attribute value into the buffer. */
1059*9663SMark.Logan@Sun.COM 	l = ntfs_attr_pread(na, 0, na->data_size, vol->upcase);
1060*9663SMark.Logan@Sun.COM 	if (l != na->data_size) {
1061*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1062*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Amount of data read does not correspond to "
1063*9663SMark.Logan@Sun.COM 				"expected length!\n");
1064*9663SMark.Logan@Sun.COM 		errno = EIO;
1065*9663SMark.Logan@Sun.COM 		goto error_exit;
1066*9663SMark.Logan@Sun.COM 	}
1067*9663SMark.Logan@Sun.COM 	/* Done with the $UpCase mft record. */
1068*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
1069*9663SMark.Logan@Sun.COM 	ntfs_attr_close(na);
1070*9663SMark.Logan@Sun.COM 	if (ntfs_inode_close(ni))
1071*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to close inode, leaking memory");
1072*9663SMark.Logan@Sun.COM 
1073*9663SMark.Logan@Sun.COM 	/*
1074*9663SMark.Logan@Sun.COM 	 * Now load $Volume and set the version information and flags in the
1075*9663SMark.Logan@Sun.COM 	 * vol structure accordingly.
1076*9663SMark.Logan@Sun.COM 	 */
1077*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Loading $Volume... ");
1078*9663SMark.Logan@Sun.COM 	vol->vol_ni = ntfs_inode_open(vol, FILE_Volume);
1079*9663SMark.Logan@Sun.COM 	if (!vol->vol_ni) {
1080*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1081*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open inode");
1082*9663SMark.Logan@Sun.COM 		goto error_exit;
1083*9663SMark.Logan@Sun.COM 	}
1084*9663SMark.Logan@Sun.COM 	/* Get a search context for the $Volume/$VOLUME_INFORMATION lookup. */
1085*9663SMark.Logan@Sun.COM 	ctx = ntfs_attr_get_search_ctx(vol->vol_ni, NULL);
1086*9663SMark.Logan@Sun.COM 	if (!ctx) {
1087*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1088*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to allocate attribute search context");
1089*9663SMark.Logan@Sun.COM 		goto error_exit;
1090*9663SMark.Logan@Sun.COM 	}
1091*9663SMark.Logan@Sun.COM 	/* Find the $VOLUME_INFORMATION attribute. */
1092*9663SMark.Logan@Sun.COM 	if (ntfs_attr_lookup(AT_VOLUME_INFORMATION, AT_UNNAMED, 0, 0, 0, NULL,
1093*9663SMark.Logan@Sun.COM 			0, ctx)) {
1094*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1095*9663SMark.Logan@Sun.COM 		ntfs_log_debug("$VOLUME_INFORMATION attribute not found in "
1096*9663SMark.Logan@Sun.COM 				"$Volume?!?\n");
1097*9663SMark.Logan@Sun.COM 		goto error_exit;
1098*9663SMark.Logan@Sun.COM 	}
1099*9663SMark.Logan@Sun.COM 	a = ctx->attr;
1100*9663SMark.Logan@Sun.COM 	/* Has to be resident. */
1101*9663SMark.Logan@Sun.COM 	if (a->non_resident) {
1102*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1103*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Error: Attribute $VOLUME_INFORMATION must be "
1104*9663SMark.Logan@Sun.COM 				"resident (and it isn't)!\n");
1105*9663SMark.Logan@Sun.COM 		errno = EIO;
1106*9663SMark.Logan@Sun.COM 		goto error_exit;
1107*9663SMark.Logan@Sun.COM 	}
1108*9663SMark.Logan@Sun.COM 	/* Get a pointer to the value of the attribute. */
1109*9663SMark.Logan@Sun.COM 	vinf = (VOLUME_INFORMATION*)(le16_to_cpu(a->u.res.value_offset) + (char*)a);
1110*9663SMark.Logan@Sun.COM 	/* Sanity checks. */
1111*9663SMark.Logan@Sun.COM 	if ((char*)vinf + le32_to_cpu(a->u.res.value_length) > (char*)ctx->mrec +
1112*9663SMark.Logan@Sun.COM 			le32_to_cpu(ctx->mrec->bytes_in_use) ||
1113*9663SMark.Logan@Sun.COM 			le16_to_cpu(a->u.res.value_offset) + le32_to_cpu(
1114*9663SMark.Logan@Sun.COM 			a->u.res.value_length) > le32_to_cpu(a->length)) {
1115*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1116*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Error: Attribute $VOLUME_INFORMATION in "
1117*9663SMark.Logan@Sun.COM 				"$Volume is corrupt!\n");
1118*9663SMark.Logan@Sun.COM 		errno = EIO;
1119*9663SMark.Logan@Sun.COM 		goto error_exit;
1120*9663SMark.Logan@Sun.COM 	}
1121*9663SMark.Logan@Sun.COM 	/* Setup vol from the volume information attribute value. */
1122*9663SMark.Logan@Sun.COM 	vol->major_ver = vinf->major_ver;
1123*9663SMark.Logan@Sun.COM 	vol->minor_ver = vinf->minor_ver;
1124*9663SMark.Logan@Sun.COM 	/*
1125*9663SMark.Logan@Sun.COM 	 * Do not use le16_to_cpu() macro here as our VOLUME_FLAGS are defined
1126*9663SMark.Logan@Sun.COM 	 * using cpu_to_le16() macro and hence are consistent.
1127*9663SMark.Logan@Sun.COM 	 */
1128*9663SMark.Logan@Sun.COM 	vol->flags = vinf->flags;
1129*9663SMark.Logan@Sun.COM 	/* Record whether the volume was dirty or not. */
1130*9663SMark.Logan@Sun.COM 	if (vol->flags & VOLUME_IS_DIRTY)
1131*9663SMark.Logan@Sun.COM 		NVolSetWasDirty(vol);
1132*9663SMark.Logan@Sun.COM 	/*
1133*9663SMark.Logan@Sun.COM 	 * Reinitialize the search context for the $Volume/$VOLUME_NAME lookup.
1134*9663SMark.Logan@Sun.COM 	 */
1135*9663SMark.Logan@Sun.COM 	ntfs_attr_reinit_search_ctx(ctx);
1136*9663SMark.Logan@Sun.COM 	if (ntfs_attr_lookup(AT_VOLUME_NAME, AT_UNNAMED, 0, 0, 0, NULL, 0,
1137*9663SMark.Logan@Sun.COM 			ctx)) {
1138*9663SMark.Logan@Sun.COM 		if (errno != ENOENT) {
1139*9663SMark.Logan@Sun.COM 			ntfs_log_debug(FAILED);
1140*9663SMark.Logan@Sun.COM 			ntfs_log_debug("Error: Lookup of $VOLUME_NAME "
1141*9663SMark.Logan@Sun.COM 					"attribute in $Volume failed.  "
1142*9663SMark.Logan@Sun.COM 					"This probably means something is "
1143*9663SMark.Logan@Sun.COM 					"corrupt.  Run chkdsk.\n");
1144*9663SMark.Logan@Sun.COM 			goto error_exit;
1145*9663SMark.Logan@Sun.COM 		}
1146*9663SMark.Logan@Sun.COM 		/*
1147*9663SMark.Logan@Sun.COM 		 * Attribute not present.  This has been seen in the field.
1148*9663SMark.Logan@Sun.COM 		 * Treat this the same way as if the attribute was present but
1149*9663SMark.Logan@Sun.COM 		 * had zero length.
1150*9663SMark.Logan@Sun.COM 		 */
1151*9663SMark.Logan@Sun.COM 		vol->vol_name = ntfs_malloc(1);
1152*9663SMark.Logan@Sun.COM 		if (!vol->vol_name) {
1153*9663SMark.Logan@Sun.COM 			ntfs_log_debug(FAILED);
1154*9663SMark.Logan@Sun.COM 			goto error_exit;
1155*9663SMark.Logan@Sun.COM 		}
1156*9663SMark.Logan@Sun.COM 		vol->vol_name[0] = '\0';
1157*9663SMark.Logan@Sun.COM 	} else {
1158*9663SMark.Logan@Sun.COM 		a = ctx->attr;
1159*9663SMark.Logan@Sun.COM 		/* Has to be resident. */
1160*9663SMark.Logan@Sun.COM 		if (a->non_resident) {
1161*9663SMark.Logan@Sun.COM 			ntfs_log_debug(FAILED);
1162*9663SMark.Logan@Sun.COM 			ntfs_log_debug("Error: Attribute $VOLUME_NAME must be "
1163*9663SMark.Logan@Sun.COM 					"resident!\n");
1164*9663SMark.Logan@Sun.COM 			errno = EIO;
1165*9663SMark.Logan@Sun.COM 			goto error_exit;
1166*9663SMark.Logan@Sun.COM 		}
1167*9663SMark.Logan@Sun.COM 		/* Get a pointer to the value of the attribute. */
1168*9663SMark.Logan@Sun.COM 		vname = (ntfschar*)(le16_to_cpu(a->u.res.value_offset) + (char*)a);
1169*9663SMark.Logan@Sun.COM 		u = le32_to_cpu(a->u.res.value_length) / 2;
1170*9663SMark.Logan@Sun.COM 		/*
1171*9663SMark.Logan@Sun.COM 		 * Convert Unicode volume name to current locale multibyte
1172*9663SMark.Logan@Sun.COM 		 * format.
1173*9663SMark.Logan@Sun.COM 		 */
1174*9663SMark.Logan@Sun.COM 		vol->vol_name = NULL;
1175*9663SMark.Logan@Sun.COM 		if (ntfs_ucstombs(vname, u, &vol->vol_name, 0) == -1) {
1176*9663SMark.Logan@Sun.COM 			ntfs_log_perror("Error: Volume name could not be "
1177*9663SMark.Logan@Sun.COM 					"converted to current locale");
1178*9663SMark.Logan@Sun.COM 			ntfs_log_debug("Forcing name into ASCII by replacing "
1179*9663SMark.Logan@Sun.COM 				"non-ASCII characters with underscores.\n");
1180*9663SMark.Logan@Sun.COM 			vol->vol_name = ntfs_malloc(u + 1);
1181*9663SMark.Logan@Sun.COM 			if (!vol->vol_name) {
1182*9663SMark.Logan@Sun.COM 				ntfs_log_debug(FAILED);
1183*9663SMark.Logan@Sun.COM 				goto error_exit;
1184*9663SMark.Logan@Sun.COM 			}
1185*9663SMark.Logan@Sun.COM 			for (j = 0; j < (s32)u; j++) {
1186*9663SMark.Logan@Sun.COM 				u16 uc = le16_to_cpu(vname[j]);
1187*9663SMark.Logan@Sun.COM 				if (uc > 0xff)
1188*9663SMark.Logan@Sun.COM 					uc = (u16)'_';
1189*9663SMark.Logan@Sun.COM 				vol->vol_name[j] = (char)uc;
1190*9663SMark.Logan@Sun.COM 			}
1191*9663SMark.Logan@Sun.COM 			vol->vol_name[u] = 0;
1192*9663SMark.Logan@Sun.COM 		}
1193*9663SMark.Logan@Sun.COM 	}
1194*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
1195*9663SMark.Logan@Sun.COM 	ntfs_attr_put_search_ctx(ctx);
1196*9663SMark.Logan@Sun.COM 	ctx = NULL;
1197*9663SMark.Logan@Sun.COM 	/* Now load the attribute definitions from $AttrDef. */
1198*9663SMark.Logan@Sun.COM 	ntfs_log_debug("Loading $AttrDef... ");
1199*9663SMark.Logan@Sun.COM 	ni = ntfs_inode_open(vol, FILE_AttrDef);
1200*9663SMark.Logan@Sun.COM 	if (!ni) {
1201*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1202*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open inode");
1203*9663SMark.Logan@Sun.COM 		goto error_exit;
1204*9663SMark.Logan@Sun.COM 	}
1205*9663SMark.Logan@Sun.COM 	/* Get an ntfs attribute for $AttrDef/$DATA. */
1206*9663SMark.Logan@Sun.COM 	na = ntfs_attr_open(ni, AT_DATA, AT_UNNAMED, 0);
1207*9663SMark.Logan@Sun.COM 	if (!na) {
1208*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1209*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open ntfs attribute");
1210*9663SMark.Logan@Sun.COM 		goto error_exit;
1211*9663SMark.Logan@Sun.COM 	}
1212*9663SMark.Logan@Sun.COM 	/* Check we don't overflow 32-bits. */
1213*9663SMark.Logan@Sun.COM 	if (na->data_size > 0xffffffffLL) {
1214*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1215*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Error: Attribute definition table is too big "
1216*9663SMark.Logan@Sun.COM 				"(max 32-bit allowed).\n");
1217*9663SMark.Logan@Sun.COM 		errno = EINVAL;
1218*9663SMark.Logan@Sun.COM 		goto error_exit;
1219*9663SMark.Logan@Sun.COM 	}
1220*9663SMark.Logan@Sun.COM 	vol->attrdef_len = na->data_size;
1221*9663SMark.Logan@Sun.COM 	vol->attrdef = (ATTR_DEF*)ntfs_malloc(na->data_size);
1222*9663SMark.Logan@Sun.COM 	if (!vol->attrdef) {
1223*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1224*9663SMark.Logan@Sun.COM 		goto error_exit;
1225*9663SMark.Logan@Sun.COM 	}
1226*9663SMark.Logan@Sun.COM 	/* Read in the $DATA attribute value into the buffer. */
1227*9663SMark.Logan@Sun.COM 	l = ntfs_attr_pread(na, 0, na->data_size, vol->attrdef);
1228*9663SMark.Logan@Sun.COM 	if (l != na->data_size) {
1229*9663SMark.Logan@Sun.COM 		ntfs_log_debug(FAILED);
1230*9663SMark.Logan@Sun.COM 		ntfs_log_debug("Amount of data read does not correspond to "
1231*9663SMark.Logan@Sun.COM 				"expected length!\n");
1232*9663SMark.Logan@Sun.COM 		errno = EIO;
1233*9663SMark.Logan@Sun.COM 		goto error_exit;
1234*9663SMark.Logan@Sun.COM 	}
1235*9663SMark.Logan@Sun.COM 	/* Done with the $AttrDef mft record. */
1236*9663SMark.Logan@Sun.COM 	ntfs_log_debug(OK);
1237*9663SMark.Logan@Sun.COM 	ntfs_attr_close(na);
1238*9663SMark.Logan@Sun.COM 	if (ntfs_inode_close(ni))
1239*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to close inode, leaking memory");
1240*9663SMark.Logan@Sun.COM 	/* Initialize number of free clusters and MFT records. */
1241*9663SMark.Logan@Sun.COM 	if (ntfs_volume_get_nr_free_mft_records(vol)) {
1242*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to calculate number of free MFTs");
1243*9663SMark.Logan@Sun.COM 		goto error_exit;
1244*9663SMark.Logan@Sun.COM 	}
1245*9663SMark.Logan@Sun.COM 	if (ntfs_volume_get_nr_free_clusters(vol)) {
1246*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to calculate number of free clusters");
1247*9663SMark.Logan@Sun.COM 		goto error_exit;
1248*9663SMark.Logan@Sun.COM 	}
1249*9663SMark.Logan@Sun.COM 	/*
1250*9663SMark.Logan@Sun.COM 	 * Check for dirty logfile and hibernated Windows.
1251*9663SMark.Logan@Sun.COM 	 * We care only about read-write mounts.
1252*9663SMark.Logan@Sun.COM 	 *
1253*9663SMark.Logan@Sun.COM 	 * If all is ok, reset the logfile and set the dirty bit on the volume.
1254*9663SMark.Logan@Sun.COM 	 *
1255*9663SMark.Logan@Sun.COM 	 * But do not do that if this is a FORENSIC mount.
1256*9663SMark.Logan@Sun.COM 	 */
1257*9663SMark.Logan@Sun.COM 	if (!(flags & NTFS_MNT_RDONLY)) {
1258*9663SMark.Logan@Sun.COM 		if (ntfs_volume_check_hiberfile(vol) < 0)
1259*9663SMark.Logan@Sun.COM 			goto error_exit;
1260*9663SMark.Logan@Sun.COM 		if (ntfs_volume_check_logfile(vol) < 0) {
1261*9663SMark.Logan@Sun.COM 			if (errno != EOPNOTSUPP || !(flags & NTFS_MNT_FORCE))
1262*9663SMark.Logan@Sun.COM 				goto error_exit;
1263*9663SMark.Logan@Sun.COM 			ntfs_log_warning("WARNING: $LogFile is not clean, "
1264*9663SMark.Logan@Sun.COM 					"forced to continue.\n");
1265*9663SMark.Logan@Sun.COM 			NVolSetWasDirty(vol); /* Leave volume dirty since we
1266*9663SMark.Logan@Sun.COM 						 empted logfile. */
1267*9663SMark.Logan@Sun.COM 		}
1268*9663SMark.Logan@Sun.COM 		if (!NVolForensicMount(vol)) {
1269*9663SMark.Logan@Sun.COM 			if (ntfs_logfile_reset(vol) < 0)
1270*9663SMark.Logan@Sun.COM 				goto error_exit;
1271*9663SMark.Logan@Sun.COM 			if (!(vol->flags & VOLUME_IS_DIRTY)) {
1272*9663SMark.Logan@Sun.COM 				vol->flags |= VOLUME_IS_DIRTY;
1273*9663SMark.Logan@Sun.COM 				if (ntfs_volume_write_flags(vol, vol->flags) <
1274*9663SMark.Logan@Sun.COM 						0)
1275*9663SMark.Logan@Sun.COM 					goto error_exit;
1276*9663SMark.Logan@Sun.COM 			}
1277*9663SMark.Logan@Sun.COM 		}
1278*9663SMark.Logan@Sun.COM 	}
1279*9663SMark.Logan@Sun.COM 	return vol;
1280*9663SMark.Logan@Sun.COM io_error_exit:
1281*9663SMark.Logan@Sun.COM 	errno = EIO;
1282*9663SMark.Logan@Sun.COM error_exit:
1283*9663SMark.Logan@Sun.COM 	eo = errno;
1284*9663SMark.Logan@Sun.COM 	if (ctx)
1285*9663SMark.Logan@Sun.COM 		ntfs_attr_put_search_ctx(ctx);
1286*9663SMark.Logan@Sun.COM 	free(m);
1287*9663SMark.Logan@Sun.COM 	free(m2);
1288*9663SMark.Logan@Sun.COM 	__ntfs_volume_release(vol);
1289*9663SMark.Logan@Sun.COM 	errno = eo;
1290*9663SMark.Logan@Sun.COM 	return NULL;
1291*9663SMark.Logan@Sun.COM }
1292*9663SMark.Logan@Sun.COM 
1293*9663SMark.Logan@Sun.COM /**
1294*9663SMark.Logan@Sun.COM  * ntfs_mount - open ntfs volume
1295*9663SMark.Logan@Sun.COM  * @name:	name of device/file to open
1296*9663SMark.Logan@Sun.COM  * @flags:	optional mount flags
1297*9663SMark.Logan@Sun.COM  *
1298*9663SMark.Logan@Sun.COM  * This function mounts an ntfs volume. @name should contain the name of the
1299*9663SMark.Logan@Sun.COM  * device/file to mount as the ntfs volume.
1300*9663SMark.Logan@Sun.COM  *
1301*9663SMark.Logan@Sun.COM  * @flags is an optional second parameter. See ntfs_device_mount comment for
1302*9663SMark.Logan@Sun.COM  * description.
1303*9663SMark.Logan@Sun.COM  *
1304*9663SMark.Logan@Sun.COM  * The function opens the device or file @name and verifies that it contains a
1305*9663SMark.Logan@Sun.COM  * valid bootsector. Then, it allocates an ntfs_volume structure and initializes
1306*9663SMark.Logan@Sun.COM  * some of the values inside the structure from the information stored in the
1307*9663SMark.Logan@Sun.COM  * bootsector. It proceeds to load the necessary system files and completes
1308*9663SMark.Logan@Sun.COM  * setting up the structure.
1309*9663SMark.Logan@Sun.COM  *
1310*9663SMark.Logan@Sun.COM  * Return the allocated volume structure on success and NULL on error with
1311*9663SMark.Logan@Sun.COM  * errno set to the error code.
1312*9663SMark.Logan@Sun.COM  *
1313*9663SMark.Logan@Sun.COM  * Note, that a copy is made of @name, and hence it can be discarded as
1314*9663SMark.Logan@Sun.COM  * soon as the function returns.
1315*9663SMark.Logan@Sun.COM  */
1316*9663SMark.Logan@Sun.COM ntfs_volume *ntfs_mount(const char *name __attribute__((unused)),
1317*9663SMark.Logan@Sun.COM 		ntfs_mount_flags flags __attribute__((unused)))
1318*9663SMark.Logan@Sun.COM {
1319*9663SMark.Logan@Sun.COM #ifndef NO_NTFS_DEVICE_DEFAULT_IO_OPS
1320*9663SMark.Logan@Sun.COM 	struct ntfs_device *dev;
1321*9663SMark.Logan@Sun.COM 	ntfs_volume *vol;
1322*9663SMark.Logan@Sun.COM 
1323*9663SMark.Logan@Sun.COM 	/* Allocate an ntfs_device structure. */
1324*9663SMark.Logan@Sun.COM 	dev = ntfs_device_alloc(name, 0, &ntfs_device_default_io_ops, NULL);
1325*9663SMark.Logan@Sun.COM 	if (!dev)
1326*9663SMark.Logan@Sun.COM 		return NULL;
1327*9663SMark.Logan@Sun.COM 	/* Call ntfs_device_mount() to do the actual mount. */
1328*9663SMark.Logan@Sun.COM 	vol = ntfs_device_mount(dev, flags);
1329*9663SMark.Logan@Sun.COM 	if (!vol) {
1330*9663SMark.Logan@Sun.COM 		int eo = errno;
1331*9663SMark.Logan@Sun.COM 		ntfs_device_free(dev);
1332*9663SMark.Logan@Sun.COM 		errno = eo;
1333*9663SMark.Logan@Sun.COM 	}
1334*9663SMark.Logan@Sun.COM 	return vol;
1335*9663SMark.Logan@Sun.COM #else
1336*9663SMark.Logan@Sun.COM 	/*
1337*9663SMark.Logan@Sun.COM 	 * ntfs_mount() makes no sense if NO_NTFS_DEVICE_DEFAULT_IO_OPS is
1338*9663SMark.Logan@Sun.COM 	 * defined as there are no device operations available in libntfs in
1339*9663SMark.Logan@Sun.COM 	 * this case.
1340*9663SMark.Logan@Sun.COM 	 */
1341*9663SMark.Logan@Sun.COM 	errno = EOPNOTSUPP;
1342*9663SMark.Logan@Sun.COM 	return NULL;
1343*9663SMark.Logan@Sun.COM #endif
1344*9663SMark.Logan@Sun.COM }
1345*9663SMark.Logan@Sun.COM 
1346*9663SMark.Logan@Sun.COM /**
1347*9663SMark.Logan@Sun.COM  * ntfs_device_umount - close ntfs volume
1348*9663SMark.Logan@Sun.COM  * @vol: address of ntfs_volume structure of volume to close
1349*9663SMark.Logan@Sun.COM  * @force: if true force close the volume even if it is busy
1350*9663SMark.Logan@Sun.COM  *
1351*9663SMark.Logan@Sun.COM  * Deallocate all structures (including @vol itself) associated with the ntfs
1352*9663SMark.Logan@Sun.COM  * volume @vol.
1353*9663SMark.Logan@Sun.COM  *
1354*9663SMark.Logan@Sun.COM  * Note it is up to the caller to destroy the device associated with the volume
1355*9663SMark.Logan@Sun.COM  * being unmounted after this function returns.
1356*9663SMark.Logan@Sun.COM  *
1357*9663SMark.Logan@Sun.COM  * Return 0 on success. On error return -1 with errno set appropriately
1358*9663SMark.Logan@Sun.COM  * (most likely to one of EAGAIN, EBUSY or EINVAL). The EAGAIN error means that
1359*9663SMark.Logan@Sun.COM  * an operation is in progress and if you try the close later the operation
1360*9663SMark.Logan@Sun.COM  * might be completed and the close succeed.
1361*9663SMark.Logan@Sun.COM  *
1362*9663SMark.Logan@Sun.COM  * If @force is true (i.e. not zero) this function will close the volume even
1363*9663SMark.Logan@Sun.COM  * if this means that data might be lost.
1364*9663SMark.Logan@Sun.COM  *
1365*9663SMark.Logan@Sun.COM  * @vol must have previously been returned by a call to ntfs_device_mount().
1366*9663SMark.Logan@Sun.COM  *
1367*9663SMark.Logan@Sun.COM  * @vol itself is deallocated and should no longer be dereferenced after this
1368*9663SMark.Logan@Sun.COM  * function returns success. If it returns an error then nothing has been done
1369*9663SMark.Logan@Sun.COM  * so it is safe to continue using @vol.
1370*9663SMark.Logan@Sun.COM  */
1371*9663SMark.Logan@Sun.COM int ntfs_device_umount(ntfs_volume *vol,
1372*9663SMark.Logan@Sun.COM 		const BOOL force __attribute__((unused)))
1373*9663SMark.Logan@Sun.COM {
1374*9663SMark.Logan@Sun.COM 	if (!vol) {
1375*9663SMark.Logan@Sun.COM 		errno = EINVAL;
1376*9663SMark.Logan@Sun.COM 		return -1;
1377*9663SMark.Logan@Sun.COM 	}
1378*9663SMark.Logan@Sun.COM 	__ntfs_volume_release(vol);
1379*9663SMark.Logan@Sun.COM 	return 0;
1380*9663SMark.Logan@Sun.COM }
1381*9663SMark.Logan@Sun.COM 
1382*9663SMark.Logan@Sun.COM /**
1383*9663SMark.Logan@Sun.COM  * ntfs_umount - close ntfs volume
1384*9663SMark.Logan@Sun.COM  * @vol: address of ntfs_volume structure of volume to close
1385*9663SMark.Logan@Sun.COM  * @force: if true force close the volume even if it is busy
1386*9663SMark.Logan@Sun.COM  *
1387*9663SMark.Logan@Sun.COM  * Deallocate all structures (including @vol itself) associated with the ntfs
1388*9663SMark.Logan@Sun.COM  * volume @vol.
1389*9663SMark.Logan@Sun.COM  *
1390*9663SMark.Logan@Sun.COM  * Return 0 on success. On error return -1 with errno set appropriately
1391*9663SMark.Logan@Sun.COM  * (most likely to one of EAGAIN, EBUSY or EINVAL). The EAGAIN error means that
1392*9663SMark.Logan@Sun.COM  * an operation is in progress and if you try the close later the operation
1393*9663SMark.Logan@Sun.COM  * might be completed and the close succeed.
1394*9663SMark.Logan@Sun.COM  *
1395*9663SMark.Logan@Sun.COM  * If @force is true (i.e. not zero) this function will close the volume even
1396*9663SMark.Logan@Sun.COM  * if this means that data might be lost.
1397*9663SMark.Logan@Sun.COM  *
1398*9663SMark.Logan@Sun.COM  * @vol must have previously been returned by a call to ntfs_mount().
1399*9663SMark.Logan@Sun.COM  *
1400*9663SMark.Logan@Sun.COM  * @vol itself is deallocated and should no longer be dereferenced after this
1401*9663SMark.Logan@Sun.COM  * function returns success. If it returns an error then nothing has been done
1402*9663SMark.Logan@Sun.COM  * so it is safe to continue using @vol.
1403*9663SMark.Logan@Sun.COM  */
1404*9663SMark.Logan@Sun.COM int ntfs_umount(ntfs_volume *vol,
1405*9663SMark.Logan@Sun.COM 		const BOOL force __attribute__((unused)))
1406*9663SMark.Logan@Sun.COM {
1407*9663SMark.Logan@Sun.COM 	struct ntfs_device *dev;
1408*9663SMark.Logan@Sun.COM 
1409*9663SMark.Logan@Sun.COM 	if (!vol) {
1410*9663SMark.Logan@Sun.COM 		errno = EINVAL;
1411*9663SMark.Logan@Sun.COM 		return -1;
1412*9663SMark.Logan@Sun.COM 	}
1413*9663SMark.Logan@Sun.COM 	dev = vol->u.dev;
1414*9663SMark.Logan@Sun.COM 	__ntfs_volume_release(vol);
1415*9663SMark.Logan@Sun.COM 	ntfs_device_free(dev);
1416*9663SMark.Logan@Sun.COM 	return 0;
1417*9663SMark.Logan@Sun.COM }
1418*9663SMark.Logan@Sun.COM 
1419*9663SMark.Logan@Sun.COM #ifdef HAVE_MNTENT_H
1420*9663SMark.Logan@Sun.COM 
1421*9663SMark.Logan@Sun.COM #ifndef HAVE_REALPATH
1422*9663SMark.Logan@Sun.COM /**
1423*9663SMark.Logan@Sun.COM  * realpath - If there is no realpath on the system
1424*9663SMark.Logan@Sun.COM  */
1425*9663SMark.Logan@Sun.COM static char *realpath(const char *path, char *resolved_path)
1426*9663SMark.Logan@Sun.COM {
1427*9663SMark.Logan@Sun.COM 	strncpy(resolved_path, path, PATH_MAX);
1428*9663SMark.Logan@Sun.COM 	resolved_path[PATH_MAX] = '\0';
1429*9663SMark.Logan@Sun.COM 	return resolved_path;
1430*9663SMark.Logan@Sun.COM }
1431*9663SMark.Logan@Sun.COM #endif
1432*9663SMark.Logan@Sun.COM 
1433*9663SMark.Logan@Sun.COM /**
1434*9663SMark.Logan@Sun.COM  * ntfs_mntent_check - desc
1435*9663SMark.Logan@Sun.COM  *
1436*9663SMark.Logan@Sun.COM  * If you are wanting to use this, you actually wanted to use
1437*9663SMark.Logan@Sun.COM  * ntfs_check_if_mounted(), you just didn't realize. (-:
1438*9663SMark.Logan@Sun.COM  *
1439*9663SMark.Logan@Sun.COM  * See description of ntfs_check_if_mounted(), below.
1440*9663SMark.Logan@Sun.COM  */
1441*9663SMark.Logan@Sun.COM static int ntfs_mntent_check(const char *file, unsigned long *mnt_flags)
1442*9663SMark.Logan@Sun.COM {
1443*9663SMark.Logan@Sun.COM 	struct mntent *mnt;
1444*9663SMark.Logan@Sun.COM 	char *real_file = NULL, *real_fsname = NULL;
1445*9663SMark.Logan@Sun.COM 	FILE *f;
1446*9663SMark.Logan@Sun.COM 	int err = 0;
1447*9663SMark.Logan@Sun.COM 
1448*9663SMark.Logan@Sun.COM 	real_file = ntfs_malloc(PATH_MAX + 1);
1449*9663SMark.Logan@Sun.COM 	if (!real_file)
1450*9663SMark.Logan@Sun.COM 		return -1;
1451*9663SMark.Logan@Sun.COM 	real_fsname = ntfs_malloc(PATH_MAX + 1);
1452*9663SMark.Logan@Sun.COM 	if (!real_fsname) {
1453*9663SMark.Logan@Sun.COM 		err = errno;
1454*9663SMark.Logan@Sun.COM 		goto exit;
1455*9663SMark.Logan@Sun.COM 	}
1456*9663SMark.Logan@Sun.COM 	if (!realpath(file, real_file)) {
1457*9663SMark.Logan@Sun.COM 		err = errno;
1458*9663SMark.Logan@Sun.COM 		goto exit;
1459*9663SMark.Logan@Sun.COM 	}
1460*9663SMark.Logan@Sun.COM 	if (!(f = setmntent(MOUNTED, "r"))) {
1461*9663SMark.Logan@Sun.COM 		err = errno;
1462*9663SMark.Logan@Sun.COM 		goto exit;
1463*9663SMark.Logan@Sun.COM 	}
1464*9663SMark.Logan@Sun.COM 	while ((mnt = getmntent(f))) {
1465*9663SMark.Logan@Sun.COM 		if (!realpath(mnt->mnt_fsname, real_fsname))
1466*9663SMark.Logan@Sun.COM 			continue;
1467*9663SMark.Logan@Sun.COM 		if (!strcmp(real_file, real_fsname))
1468*9663SMark.Logan@Sun.COM 			break;
1469*9663SMark.Logan@Sun.COM 	}
1470*9663SMark.Logan@Sun.COM 	endmntent(f);
1471*9663SMark.Logan@Sun.COM 	if (!mnt)
1472*9663SMark.Logan@Sun.COM 		goto exit;
1473*9663SMark.Logan@Sun.COM 	*mnt_flags = NTFS_MF_MOUNTED;
1474*9663SMark.Logan@Sun.COM 	if (!strcmp(mnt->mnt_dir, "/"))
1475*9663SMark.Logan@Sun.COM 		*mnt_flags |= NTFS_MF_ISROOT;
1476*9663SMark.Logan@Sun.COM #ifdef HAVE_HASMNTOPT
1477*9663SMark.Logan@Sun.COM 	if (hasmntopt(mnt, "ro") && !hasmntopt(mnt, "rw"))
1478*9663SMark.Logan@Sun.COM 		*mnt_flags |= NTFS_MF_READONLY;
1479*9663SMark.Logan@Sun.COM #endif
1480*9663SMark.Logan@Sun.COM exit:
1481*9663SMark.Logan@Sun.COM 	free(real_file);
1482*9663SMark.Logan@Sun.COM 	free(real_fsname);
1483*9663SMark.Logan@Sun.COM 	if (err) {
1484*9663SMark.Logan@Sun.COM 		errno = err;
1485*9663SMark.Logan@Sun.COM 		return -1;
1486*9663SMark.Logan@Sun.COM 	}
1487*9663SMark.Logan@Sun.COM 	return 0;
1488*9663SMark.Logan@Sun.COM }
1489*9663SMark.Logan@Sun.COM #endif /* HAVE_MNTENT_H */
1490*9663SMark.Logan@Sun.COM 
1491*9663SMark.Logan@Sun.COM /**
1492*9663SMark.Logan@Sun.COM  * ntfs_check_if_mounted - check if an ntfs volume is currently mounted
1493*9663SMark.Logan@Sun.COM  * @file:	device file to check
1494*9663SMark.Logan@Sun.COM  * @mnt_flags:	pointer into which to return the ntfs mount flags (see volume.h)
1495*9663SMark.Logan@Sun.COM  *
1496*9663SMark.Logan@Sun.COM  * If the running system does not support the {set,get,end}mntent() calls,
1497*9663SMark.Logan@Sun.COM  * just return 0 and set *@mnt_flags to zero.
1498*9663SMark.Logan@Sun.COM  *
1499*9663SMark.Logan@Sun.COM  * When the system does support the calls, ntfs_check_if_mounted() first tries
1500*9663SMark.Logan@Sun.COM  * to find the device @file in /etc/mtab (or wherever this is kept on the
1501*9663SMark.Logan@Sun.COM  * running system). If it is not found, assume the device is not mounted and
1502*9663SMark.Logan@Sun.COM  * return 0 and set *@mnt_flags to zero.
1503*9663SMark.Logan@Sun.COM  *
1504*9663SMark.Logan@Sun.COM  * If the device @file is found, set the NTFS_MF_MOUNTED flags in *@mnt_flags.
1505*9663SMark.Logan@Sun.COM  *
1506*9663SMark.Logan@Sun.COM  * Further if @file is mounted as the file system root ("/"), set the flag
1507*9663SMark.Logan@Sun.COM  * NTFS_MF_ISROOT in *@mnt_flags.
1508*9663SMark.Logan@Sun.COM  *
1509*9663SMark.Logan@Sun.COM  * Finally, check if the file system is mounted read-only, and if so set the
1510*9663SMark.Logan@Sun.COM  * NTFS_MF_READONLY flag in *@mnt_flags.
1511*9663SMark.Logan@Sun.COM  *
1512*9663SMark.Logan@Sun.COM  * On success return 0 with *@mnt_flags set to the ntfs mount flags.
1513*9663SMark.Logan@Sun.COM  *
1514*9663SMark.Logan@Sun.COM  * On error return -1 with errno set to the error code.
1515*9663SMark.Logan@Sun.COM  */
1516*9663SMark.Logan@Sun.COM int ntfs_check_if_mounted(const char *file __attribute__((unused)),
1517*9663SMark.Logan@Sun.COM 		unsigned long *mnt_flags)
1518*9663SMark.Logan@Sun.COM {
1519*9663SMark.Logan@Sun.COM 	*mnt_flags = 0;
1520*9663SMark.Logan@Sun.COM #ifdef HAVE_MNTENT_H
1521*9663SMark.Logan@Sun.COM 	return ntfs_mntent_check(file, mnt_flags);
1522*9663SMark.Logan@Sun.COM #else
1523*9663SMark.Logan@Sun.COM 	return 0;
1524*9663SMark.Logan@Sun.COM #endif
1525*9663SMark.Logan@Sun.COM }
1526*9663SMark.Logan@Sun.COM 
1527*9663SMark.Logan@Sun.COM /**
1528*9663SMark.Logan@Sun.COM  * ntfs_version_is_supported - check if NTFS version is supported.
1529*9663SMark.Logan@Sun.COM  * @vol:	ntfs volume whose version we're interested in.
1530*9663SMark.Logan@Sun.COM  *
1531*9663SMark.Logan@Sun.COM  * The function checks if the NTFS volume version is known or not.
1532*9663SMark.Logan@Sun.COM  * Version 1.1 and 1.2 are used by Windows NT3.x and NT4.
1533*9663SMark.Logan@Sun.COM  * Version 2.x is used by Windows 2000 Betas.
1534*9663SMark.Logan@Sun.COM  * Version 3.0 is used by Windows 2000.
1535*9663SMark.Logan@Sun.COM  * Version 3.1 is used by Windows XP, Windows Server 2003 and Vista.
1536*9663SMark.Logan@Sun.COM  *
1537*9663SMark.Logan@Sun.COM  * Return 0 if NTFS version is supported otherwise -1 with errno set.
1538*9663SMark.Logan@Sun.COM  *
1539*9663SMark.Logan@Sun.COM  * The following error codes are defined:
1540*9663SMark.Logan@Sun.COM  *	EOPNOTSUPP - Unknown NTFS version
1541*9663SMark.Logan@Sun.COM  *	EINVAL	   - Invalid argument
1542*9663SMark.Logan@Sun.COM  */
1543*9663SMark.Logan@Sun.COM int ntfs_version_is_supported(ntfs_volume *vol)
1544*9663SMark.Logan@Sun.COM {
1545*9663SMark.Logan@Sun.COM 	u8 major, minor;
1546*9663SMark.Logan@Sun.COM 
1547*9663SMark.Logan@Sun.COM 	if (!vol) {
1548*9663SMark.Logan@Sun.COM 		errno = EINVAL;
1549*9663SMark.Logan@Sun.COM 		return -1;
1550*9663SMark.Logan@Sun.COM 	}
1551*9663SMark.Logan@Sun.COM 
1552*9663SMark.Logan@Sun.COM 	major = vol->major_ver;
1553*9663SMark.Logan@Sun.COM 	minor = vol->minor_ver;
1554*9663SMark.Logan@Sun.COM 
1555*9663SMark.Logan@Sun.COM 	if (NTFS_V1_1(major, minor) || NTFS_V1_2(major, minor))
1556*9663SMark.Logan@Sun.COM 		return 0;
1557*9663SMark.Logan@Sun.COM 
1558*9663SMark.Logan@Sun.COM 	if (NTFS_V2_X(major, minor))
1559*9663SMark.Logan@Sun.COM 		return 0;
1560*9663SMark.Logan@Sun.COM 
1561*9663SMark.Logan@Sun.COM 	if (NTFS_V3_0(major, minor) || NTFS_V3_1(major, minor))
1562*9663SMark.Logan@Sun.COM 		return 0;
1563*9663SMark.Logan@Sun.COM 
1564*9663SMark.Logan@Sun.COM 	errno = EOPNOTSUPP;
1565*9663SMark.Logan@Sun.COM 	return -1;
1566*9663SMark.Logan@Sun.COM }
1567*9663SMark.Logan@Sun.COM 
1568*9663SMark.Logan@Sun.COM /**
1569*9663SMark.Logan@Sun.COM  * ntfs_logfile_reset - "empty" $LogFile data attribute value
1570*9663SMark.Logan@Sun.COM  * @vol:	ntfs volume whose $LogFile we intend to reset.
1571*9663SMark.Logan@Sun.COM  *
1572*9663SMark.Logan@Sun.COM  * Fill the value of the $LogFile data attribute, i.e. the contents of
1573*9663SMark.Logan@Sun.COM  * the file, with 0xff's, thus marking the journal as empty.
1574*9663SMark.Logan@Sun.COM  *
1575*9663SMark.Logan@Sun.COM  * FIXME(?): We might need to zero the LSN field of every single mft
1576*9663SMark.Logan@Sun.COM  * record as well. (But, first try without doing that and see what
1577*9663SMark.Logan@Sun.COM  * happens, since chkdsk might pickup the pieces and do it for us...)
1578*9663SMark.Logan@Sun.COM  *
1579*9663SMark.Logan@Sun.COM  * On success return 0.
1580*9663SMark.Logan@Sun.COM  *
1581*9663SMark.Logan@Sun.COM  * On error return -1 with errno set to the error code.
1582*9663SMark.Logan@Sun.COM  */
1583*9663SMark.Logan@Sun.COM int ntfs_logfile_reset(ntfs_volume *vol)
1584*9663SMark.Logan@Sun.COM {
1585*9663SMark.Logan@Sun.COM 	ntfs_inode *ni;
1586*9663SMark.Logan@Sun.COM 	ntfs_attr *na;
1587*9663SMark.Logan@Sun.COM 	int eo;
1588*9663SMark.Logan@Sun.COM 
1589*9663SMark.Logan@Sun.COM 	if (!vol) {
1590*9663SMark.Logan@Sun.COM 		errno = EINVAL;
1591*9663SMark.Logan@Sun.COM 		return -1;
1592*9663SMark.Logan@Sun.COM 	}
1593*9663SMark.Logan@Sun.COM 
1594*9663SMark.Logan@Sun.COM 	if ((ni = ntfs_inode_open(vol, FILE_LogFile)) == NULL) {
1595*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open inode FILE_LogFile.");
1596*9663SMark.Logan@Sun.COM 		return -1;
1597*9663SMark.Logan@Sun.COM 	}
1598*9663SMark.Logan@Sun.COM 
1599*9663SMark.Logan@Sun.COM 	if ((na = ntfs_attr_open(ni, AT_DATA, AT_UNNAMED, 0)) == NULL) {
1600*9663SMark.Logan@Sun.COM 		eo = errno;
1601*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to open $FILE_LogFile/$DATA");
1602*9663SMark.Logan@Sun.COM 		goto error_exit;
1603*9663SMark.Logan@Sun.COM 	}
1604*9663SMark.Logan@Sun.COM 
1605*9663SMark.Logan@Sun.COM 	if (ntfs_empty_logfile(na)) {
1606*9663SMark.Logan@Sun.COM 		eo = errno;
1607*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to empty $FILE_LogFile/$DATA");
1608*9663SMark.Logan@Sun.COM 		ntfs_attr_close(na);
1609*9663SMark.Logan@Sun.COM 		goto error_exit;
1610*9663SMark.Logan@Sun.COM 	}
1611*9663SMark.Logan@Sun.COM 	ntfs_attr_close(na);
1612*9663SMark.Logan@Sun.COM 	return ntfs_inode_close(ni);
1613*9663SMark.Logan@Sun.COM 
1614*9663SMark.Logan@Sun.COM error_exit:
1615*9663SMark.Logan@Sun.COM 	ntfs_inode_close(ni);
1616*9663SMark.Logan@Sun.COM 	errno = eo;
1617*9663SMark.Logan@Sun.COM 	return -1;
1618*9663SMark.Logan@Sun.COM }
1619*9663SMark.Logan@Sun.COM 
1620*9663SMark.Logan@Sun.COM /**
1621*9663SMark.Logan@Sun.COM  * ntfs_volume_write_flags - set the flags of an ntfs volume
1622*9663SMark.Logan@Sun.COM  * @vol:	ntfs volume where we set the volume flags
1623*9663SMark.Logan@Sun.COM  * @flags:	new flags
1624*9663SMark.Logan@Sun.COM  *
1625*9663SMark.Logan@Sun.COM  * Set the on-disk volume flags in the mft record of $Volume and
1626*9663SMark.Logan@Sun.COM  * on volume @vol to @flags.
1627*9663SMark.Logan@Sun.COM  *
1628*9663SMark.Logan@Sun.COM  * Return 0 if successful and -1 if not with errno set to the error code.
1629*9663SMark.Logan@Sun.COM  */
1630*9663SMark.Logan@Sun.COM int ntfs_volume_write_flags(ntfs_volume *vol, const le16 flags)
1631*9663SMark.Logan@Sun.COM {
1632*9663SMark.Logan@Sun.COM 	ATTR_RECORD *a;
1633*9663SMark.Logan@Sun.COM 	VOLUME_INFORMATION *c;
1634*9663SMark.Logan@Sun.COM 	ntfs_attr_search_ctx *ctx;
1635*9663SMark.Logan@Sun.COM 	int ret = -1;	/* failure */
1636*9663SMark.Logan@Sun.COM 
1637*9663SMark.Logan@Sun.COM 	if (!vol || !vol->vol_ni) {
1638*9663SMark.Logan@Sun.COM 		errno = EINVAL;
1639*9663SMark.Logan@Sun.COM 		return -1;
1640*9663SMark.Logan@Sun.COM 	}
1641*9663SMark.Logan@Sun.COM 	/* Get a pointer to the volume information attribute. */
1642*9663SMark.Logan@Sun.COM 	ctx = ntfs_attr_get_search_ctx(vol->vol_ni, NULL);
1643*9663SMark.Logan@Sun.COM 	if (!ctx) {
1644*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Failed to allocate attribute search context");
1645*9663SMark.Logan@Sun.COM 		return -1;
1646*9663SMark.Logan@Sun.COM 	}
1647*9663SMark.Logan@Sun.COM 	if (ntfs_attr_lookup(AT_VOLUME_INFORMATION, AT_UNNAMED, 0, 0, 0, NULL,
1648*9663SMark.Logan@Sun.COM 			0, ctx)) {
1649*9663SMark.Logan@Sun.COM 		ntfs_log_error("Attribute $VOLUME_INFORMATION was not found "
1650*9663SMark.Logan@Sun.COM 				"in $Volume!\n");
1651*9663SMark.Logan@Sun.COM 		goto err_out;
1652*9663SMark.Logan@Sun.COM 	}
1653*9663SMark.Logan@Sun.COM 	a = ctx->attr;
1654*9663SMark.Logan@Sun.COM 	/* Sanity check. */
1655*9663SMark.Logan@Sun.COM 	if (a->non_resident) {
1656*9663SMark.Logan@Sun.COM 		ntfs_log_error("Attribute $VOLUME_INFORMATION must be "
1657*9663SMark.Logan@Sun.COM 				"resident (and it isn't)!\n");
1658*9663SMark.Logan@Sun.COM 		errno = EIO;
1659*9663SMark.Logan@Sun.COM 		goto err_out;
1660*9663SMark.Logan@Sun.COM 	}
1661*9663SMark.Logan@Sun.COM 	/* Get a pointer to the value of the attribute. */
1662*9663SMark.Logan@Sun.COM 	c = (VOLUME_INFORMATION*)(le16_to_cpu(a->u.res.value_offset) + (char*)a);
1663*9663SMark.Logan@Sun.COM 	/* Sanity checks. */
1664*9663SMark.Logan@Sun.COM 	if ((char*)c + le32_to_cpu(a->u.res.value_length) > (char*)ctx->mrec +
1665*9663SMark.Logan@Sun.COM 			le32_to_cpu(ctx->mrec->bytes_in_use) ||
1666*9663SMark.Logan@Sun.COM 			le16_to_cpu(a->u.res.value_offset) +
1667*9663SMark.Logan@Sun.COM 			le32_to_cpu(a->u.res.value_length) > le32_to_cpu(a->length)) {
1668*9663SMark.Logan@Sun.COM 		ntfs_log_error("Attribute $VOLUME_INFORMATION in $Volume is "
1669*9663SMark.Logan@Sun.COM 				"corrupt!\n");
1670*9663SMark.Logan@Sun.COM 		errno = EIO;
1671*9663SMark.Logan@Sun.COM 		goto err_out;
1672*9663SMark.Logan@Sun.COM 	}
1673*9663SMark.Logan@Sun.COM 	/* Set the volume flags. */
1674*9663SMark.Logan@Sun.COM 	vol->flags = c->flags = flags & VOLUME_FLAGS_MASK;
1675*9663SMark.Logan@Sun.COM 	/* Write them to disk. */
1676*9663SMark.Logan@Sun.COM 	ntfs_inode_mark_dirty(vol->vol_ni);
1677*9663SMark.Logan@Sun.COM 	if (ntfs_inode_sync(vol->vol_ni)) {
1678*9663SMark.Logan@Sun.COM 		ntfs_log_perror("Error writing $Volume");
1679*9663SMark.Logan@Sun.COM 		goto err_out;
1680*9663SMark.Logan@Sun.COM 	}
1681*9663SMark.Logan@Sun.COM 	ret = 0; /* success */
1682*9663SMark.Logan@Sun.COM err_out:
1683*9663SMark.Logan@Sun.COM 	ntfs_attr_put_search_ctx(ctx);
1684*9663SMark.Logan@Sun.COM 	if (ret)
1685*9663SMark.Logan@Sun.COM 		ntfs_log_error("%s(): Failed.\n", "ntfs_volume_write_flags");
1686*9663SMark.Logan@Sun.COM 	return ret;
1687*9663SMark.Logan@Sun.COM }
1688*9663SMark.Logan@Sun.COM 
1689