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