| /** |
| * inode.c - Inode handling code. Originated from the Linux-NTFS project. |
| * |
| * Copyright (c) 2002-2005 Anton Altaparmakov |
| * Copyright (c) 2002-2008 Szabolcs Szakacsits |
| * Copyright (c) 2004-2007 Yura Pakhuchiy |
| * Copyright (c) 2004-2005 Richard Russon |
| * Copyright (c) 2009-2010 Jean-Pierre Andre |
| * |
| * This program/include file is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License as published |
| * by the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program/include file is distributed in the hope that it will be |
| * useful, but WITHOUT ANY WARRANTY; without even the implied warranty |
| * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program (in the main directory of the NTFS-3G |
| * distribution in the file COPYING); if not, write to the Free Software |
| * Foundation,Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include "config.h" |
| #endif |
| |
| #ifdef HAVE_STDLIB_H |
| #include <stdlib.h> |
| #endif |
| #ifdef HAVE_STRING_H |
| #include <string.h> |
| #endif |
| #ifdef HAVE_ERRNO_H |
| #include <errno.h> |
| #endif |
| #ifdef HAVE_SETXATTR |
| #include <sys/xattr.h> |
| #endif |
| |
| #include "param.h" |
| #include "compat.h" |
| #include "types.h" |
| #include "volume.h" |
| #include "cache.h" |
| #include "inode.h" |
| #include "attrib.h" |
| #include "debug.h" |
| #include "mft.h" |
| #include "attrlist.h" |
| #include "runlist.h" |
| #include "lcnalloc.h" |
| #include "index.h" |
| #include "dir.h" |
| #include "ntfstime.h" |
| #include "logging.h" |
| #include "misc.h" |
| |
| ntfs_inode *ntfs_inode_base(ntfs_inode *ni) |
| { |
| if (ni->nr_extents == -1) |
| return ni->base_ni; |
| return ni; |
| } |
| |
| /** |
| * ntfs_inode_mark_dirty - set the inode (and its base inode if it exists) dirty |
| * @ni: ntfs inode to set dirty |
| * |
| * Set the inode @ni dirty so it is written out later (at the latest at |
| * ntfs_inode_close() time). If @ni is an extent inode, set the base inode |
| * dirty, too. |
| * |
| * This function cannot fail. |
| */ |
| void ntfs_inode_mark_dirty(ntfs_inode *ni) |
| { |
| NInoSetDirty(ni); |
| if (ni->nr_extents == -1) |
| NInoSetDirty(ni->base_ni); |
| } |
| |
| /** |
| * __ntfs_inode_allocate - Create and initialise an NTFS inode object |
| * @vol: |
| * |
| * Description... |
| * |
| * Returns: |
| */ |
| static ntfs_inode *__ntfs_inode_allocate(ntfs_volume *vol) |
| { |
| ntfs_inode *ni; |
| |
| ni = (ntfs_inode*)ntfs_calloc(sizeof(ntfs_inode)); |
| if (ni) |
| ni->vol = vol; |
| return ni; |
| } |
| |
| /** |
| * ntfs_inode_allocate - Create an NTFS inode object |
| * @vol: |
| * |
| * Description... |
| * |
| * Returns: |
| */ |
| ntfs_inode *ntfs_inode_allocate(ntfs_volume *vol) |
| { |
| return __ntfs_inode_allocate(vol); |
| } |
| |
| /** |
| * __ntfs_inode_release - Destroy an NTFS inode object |
| * @ni: |
| * |
| * Description... |
| * |
| * Returns: |
| */ |
| static void __ntfs_inode_release(ntfs_inode *ni) |
| { |
| if (NInoDirty(ni)) |
| ntfs_log_error("Releasing dirty inode %lld!\n", |
| (long long)ni->mft_no); |
| if (NInoAttrList(ni) && ni->attr_list) |
| free(ni->attr_list); |
| free(ni->mrec); |
| free(ni); |
| return; |
| } |
| |
| /** |
| * ntfs_inode_open - open an inode ready for access |
| * @vol: volume to get the inode from |
| * @mref: inode number / mft record number to open |
| * |
| * Allocate an ntfs_inode structure and initialize it for the given inode |
| * specified by @mref. @mref specifies the inode number / mft record to read, |
| * including the sequence number, which can be 0 if no sequence number checking |
| * is to be performed. |
| * |
| * Then, allocate a buffer for the mft record, read the mft record from the |
| * volume @vol, and attach it to the ntfs_inode structure (->mrec). The |
| * mft record is mst deprotected and sanity checked for validity and we abort |
| * if deprotection or checks fail. |
| * |
| * Finally, search for an attribute list attribute in the mft record and if one |
| * is found, load the attribute list attribute value and attach it to the |
| * ntfs_inode structure (->attr_list). Also set the NI_AttrList bit to indicate |
| * this. |
| * |
| * Return a pointer to the ntfs_inode structure on success or NULL on error, |
| * with errno set to the error code. |
| */ |
| static ntfs_inode *ntfs_inode_real_open(ntfs_volume *vol, const MFT_REF mref) |
| { |
| s64 l; |
| ntfs_inode *ni = NULL; |
| ntfs_attr_search_ctx *ctx; |
| STANDARD_INFORMATION *std_info; |
| le32 lthle; |
| int olderrno; |
| |
| ntfs_log_enter("Entering for inode %lld\n", (long long)MREF(mref)); |
| if (!vol) { |
| errno = EINVAL; |
| goto out; |
| } |
| ni = __ntfs_inode_allocate(vol); |
| if (!ni) |
| goto out; |
| if (ntfs_file_record_read(vol, mref, &ni->mrec, NULL)) |
| goto err_out; |
| if (!(ni->mrec->flags & MFT_RECORD_IN_USE)) { |
| errno = ENOENT; |
| goto err_out; |
| } |
| ni->mft_no = MREF(mref); |
| ctx = ntfs_attr_get_search_ctx(ni, NULL); |
| if (!ctx) |
| goto err_out; |
| /* Receive some basic information about inode. */ |
| if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, AT_UNNAMED, |
| 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { |
| if (!ni->mrec->base_mft_record) |
| ntfs_log_perror("No STANDARD_INFORMATION in base record" |
| " %lld", (long long)MREF(mref)); |
| goto put_err_out; |
| } |
| std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->value_offset)); |
| ni->flags = std_info->file_attributes; |
| ni->creation_time = std_info->creation_time; |
| ni->last_data_change_time = std_info->last_data_change_time; |
| ni->last_mft_change_time = std_info->last_mft_change_time; |
| ni->last_access_time = std_info->last_access_time; |
| /* JPA insert v3 extensions if present */ |
| /* length may be seen as 72 (v1.x) or 96 (v3.x) */ |
| lthle = ctx->attr->length; |
| if (le32_to_cpu(lthle) > sizeof(STANDARD_INFORMATION)) { |
| set_nino_flag(ni, v3_Extensions); |
| ni->owner_id = std_info->owner_id; |
| ni->security_id = std_info->security_id; |
| ni->quota_charged = std_info->quota_charged; |
| ni->usn = std_info->usn; |
| } else { |
| clear_nino_flag(ni, v3_Extensions); |
| ni->owner_id = const_cpu_to_le32(0); |
| ni->security_id = const_cpu_to_le32(0); |
| } |
| /* Set attribute list information. */ |
| olderrno = errno; |
| if (ntfs_attr_lookup(AT_ATTRIBUTE_LIST, AT_UNNAMED, 0, |
| CASE_SENSITIVE, 0, NULL, 0, ctx)) { |
| if (errno != ENOENT) |
| goto put_err_out; |
| /* Attribute list attribute does not present. */ |
| /* restore previous errno to avoid misinterpretation */ |
| errno = olderrno; |
| goto get_size; |
| } |
| NInoSetAttrList(ni); |
| l = ntfs_get_attribute_value_length(ctx->attr); |
| if (!l) |
| goto put_err_out; |
| if (l > 0x40000) { |
| errno = EIO; |
| ntfs_log_perror("Too large attrlist attribute (%lld), inode " |
| "%lld", (long long)l, (long long)MREF(mref)); |
| goto put_err_out; |
| } |
| ni->attr_list_size = l; |
| ni->attr_list = ntfs_malloc(ni->attr_list_size); |
| if (!ni->attr_list) |
| goto put_err_out; |
| l = ntfs_get_attribute_value(vol, ctx->attr, ni->attr_list); |
| if (!l) |
| goto put_err_out; |
| if (l != ni->attr_list_size) { |
| errno = EIO; |
| ntfs_log_perror("Unexpected attrlist size (%lld <> %u), inode " |
| "%lld", (long long)l, ni->attr_list_size, |
| (long long)MREF(mref)); |
| goto put_err_out; |
| } |
| get_size: |
| olderrno = errno; |
| if (ntfs_attr_lookup(AT_DATA, AT_UNNAMED, 0, 0, 0, NULL, 0, ctx)) { |
| if (errno != ENOENT) |
| goto put_err_out; |
| /* Directory or special file. */ |
| /* restore previous errno to avoid misinterpretation */ |
| errno = olderrno; |
| ni->data_size = ni->allocated_size = 0; |
| } else { |
| if (ctx->attr->non_resident) { |
| ni->data_size = sle64_to_cpu(ctx->attr->data_size); |
| if (ctx->attr->flags & |
| (ATTR_IS_COMPRESSED | ATTR_IS_SPARSE)) |
| ni->allocated_size = sle64_to_cpu( |
| ctx->attr->compressed_size); |
| else |
| ni->allocated_size = sle64_to_cpu( |
| ctx->attr->allocated_size); |
| } else { |
| ni->data_size = le32_to_cpu(ctx->attr->value_length); |
| ni->allocated_size = (ni->data_size + 7) & ~7; |
| } |
| set_nino_flag(ni,KnownSize); |
| } |
| ntfs_attr_put_search_ctx(ctx); |
| out: |
| ntfs_log_leave("\n"); |
| return ni; |
| |
| put_err_out: |
| ntfs_attr_put_search_ctx(ctx); |
| err_out: |
| __ntfs_inode_release(ni); |
| ni = NULL; |
| goto out; |
| } |
| |
| /** |
| * ntfs_inode_close - close an ntfs inode and free all associated memory |
| * @ni: ntfs inode to close |
| * |
| * Make sure the ntfs inode @ni is clean. |
| * |
| * If the ntfs inode @ni is a base inode, close all associated extent inodes, |
| * then deallocate all memory attached to it, and finally free the ntfs inode |
| * structure itself. |
| * |
| * If it is an extent inode, we disconnect it from its base inode before we |
| * destroy it. |
| * |
| * It is OK to pass NULL to this function, it is just noop in this case. |
| * |
| * Return 0 on success or -1 on error with errno set to the error code. On |
| * error, @ni has not been freed. The user should attempt to handle the error |
| * and call ntfs_inode_close() again. The following error codes are defined: |
| * |
| * EBUSY @ni and/or its attribute list runlist is/are dirty and the |
| * attempt to write it/them to disk failed. |
| * EINVAL @ni is invalid (probably it is an extent inode). |
| * EIO I/O error while trying to write inode to disk. |
| */ |
| |
| int ntfs_inode_real_close(ntfs_inode *ni) |
| { |
| int ret = -1; |
| |
| if (!ni) |
| return 0; |
| |
| ntfs_log_enter("Entering for inode %lld\n", (long long)ni->mft_no); |
| |
| /* If we have dirty metadata, write it out. */ |
| if (NInoDirty(ni) || NInoAttrListDirty(ni)) { |
| if (ntfs_inode_sync(ni)) { |
| if (errno != EIO) |
| errno = EBUSY; |
| goto err; |
| } |
| } |
| /* Is this a base inode with mapped extent inodes? */ |
| if (ni->nr_extents > 0) { |
| while (ni->nr_extents > 0) { |
| if (ntfs_inode_real_close(ni->extent_nis[0])) { |
| if (errno != EIO) |
| errno = EBUSY; |
| goto err; |
| } |
| } |
| } else if (ni->nr_extents == -1) { |
| ntfs_inode **tmp_nis; |
| ntfs_inode *base_ni; |
| s32 i; |
| |
| /* |
| * If the inode is an extent inode, disconnect it from the |
| * base inode before destroying it. |
| */ |
| base_ni = ni->base_ni; |
| for (i = 0; i < base_ni->nr_extents; ++i) { |
| tmp_nis = base_ni->extent_nis; |
| if (tmp_nis[i] != ni) |
| continue; |
| /* Found it. Disconnect. */ |
| memmove(tmp_nis + i, tmp_nis + i + 1, |
| (base_ni->nr_extents - i - 1) * |
| sizeof(ntfs_inode *)); |
| /* Buffer should be for multiple of four extents. */ |
| if ((--base_ni->nr_extents) & 3) { |
| i = -1; |
| break; |
| } |
| /* |
| * ElectricFence is unhappy with realloc(x,0) as free(x) |
| * thus we explicitly separate these two cases. |
| */ |
| if (base_ni->nr_extents) { |
| /* Resize the memory buffer. */ |
| tmp_nis = realloc(tmp_nis, base_ni->nr_extents * |
| sizeof(ntfs_inode *)); |
| /* Ignore errors, they don't really matter. */ |
| if (tmp_nis) |
| base_ni->extent_nis = tmp_nis; |
| } else if (tmp_nis) { |
| free(tmp_nis); |
| base_ni->extent_nis = (ntfs_inode**)NULL; |
| } |
| /* Allow for error checking. */ |
| i = -1; |
| break; |
| } |
| |
| /* |
| * We could successfully sync, so only log this error |
| * and try to sync other inode extents too. |
| */ |
| if (i != -1) |
| ntfs_log_error("Extent inode %lld was not found\n", |
| (long long)ni->mft_no); |
| } |
| |
| __ntfs_inode_release(ni); |
| ret = 0; |
| err: |
| ntfs_log_leave("\n"); |
| return ret; |
| } |
| |
| #if CACHE_NIDATA_SIZE |
| |
| /* |
| * Free an inode structure when there is not more space |
| * in the cache |
| */ |
| |
| void ntfs_inode_nidata_free(const struct CACHED_GENERIC *cached) |
| { |
| ntfs_inode_real_close(((const struct CACHED_NIDATA*)cached)->ni); |
| } |
| |
| /* |
| * Compute a hash value for an inode entry |
| */ |
| |
| int ntfs_inode_nidata_hash(const struct CACHED_GENERIC *item) |
| { |
| return (((const struct CACHED_NIDATA*)item)->inum |
| % (2*CACHE_NIDATA_SIZE)); |
| } |
| |
| /* |
| * inum comparing for entering/fetching from cache |
| */ |
| |
| static int idata_cache_compare(const struct CACHED_GENERIC *cached, |
| const struct CACHED_GENERIC *wanted) |
| { |
| return (((const struct CACHED_NIDATA*)cached)->inum |
| != ((const struct CACHED_NIDATA*)wanted)->inum); |
| } |
| |
| /* |
| * Invalidate an inode entry when not needed anymore. |
| * The entry should have been synced, it may be reused later, |
| * if it is requested before it is dropped from cache. |
| */ |
| |
| void ntfs_inode_invalidate(ntfs_volume *vol, const MFT_REF mref) |
| { |
| struct CACHED_NIDATA item; |
| |
| item.inum = MREF(mref); |
| item.ni = (ntfs_inode*)NULL; |
| item.pathname = (const char*)NULL; |
| item.varsize = 0; |
| ntfs_invalidate_cache(vol->nidata_cache, |
| GENERIC(&item),idata_cache_compare,CACHE_FREE); |
| } |
| |
| #endif |
| |
| /* |
| * Open an inode |
| * |
| * When possible, an entry recorded in the cache is reused |
| * |
| * **NEVER REOPEN** an inode, this can lead to a duplicated |
| * cache entry (hard to detect), and to an obsolete one being |
| * reused. System files are however protected from being cached. |
| */ |
| |
| ntfs_inode *ntfs_inode_open(ntfs_volume *vol, const MFT_REF mref) |
| { |
| ntfs_inode *ni; |
| #if CACHE_NIDATA_SIZE |
| struct CACHED_NIDATA item; |
| struct CACHED_NIDATA *cached; |
| |
| /* fetch idata from cache */ |
| item.inum = MREF(mref); |
| debug_double_inode(item.inum,1); |
| item.pathname = (const char*)NULL; |
| item.varsize = 0; |
| cached = (struct CACHED_NIDATA*)ntfs_fetch_cache(vol->nidata_cache, |
| GENERIC(&item),idata_cache_compare); |
| if (cached) { |
| ni = cached->ni; |
| /* do not keep open entries in cache */ |
| ntfs_remove_cache(vol->nidata_cache, |
| (struct CACHED_GENERIC*)cached,0); |
| } else { |
| ni = ntfs_inode_real_open(vol, mref); |
| } |
| if (!ni) { |
| debug_double_inode(item.inum, 0); |
| } |
| #else |
| ni = ntfs_inode_real_open(vol, mref); |
| #endif |
| return (ni); |
| } |
| |
| /* |
| * Close an inode entry |
| * |
| * If cacheing is in use, the entry is synced and kept available |
| * in cache for further use. |
| * |
| * System files (inode < 16 or having the IS_4 flag) are protected |
| * against being cached. |
| */ |
| |
| int ntfs_inode_close(ntfs_inode *ni) |
| { |
| int res; |
| #if CACHE_NIDATA_SIZE |
| BOOL dirty; |
| struct CACHED_NIDATA item; |
| |
| if (ni) { |
| debug_double_inode(ni->mft_no,0); |
| /* do not cache system files : could lead to double entries */ |
| if (ni->vol && ni->vol->nidata_cache |
| && ((ni->mft_no == FILE_root) |
| || ((ni->mft_no >= FILE_first_user) |
| && !(ni->mrec->flags & MFT_RECORD_IS_4)))) { |
| /* If we have dirty metadata, write it out. */ |
| dirty = NInoDirty(ni) || NInoAttrListDirty(ni); |
| if (dirty) { |
| res = ntfs_inode_sync(ni); |
| /* do a real close if sync failed */ |
| if (res) |
| ntfs_inode_real_close(ni); |
| } else |
| res = 0; |
| |
| if (!res) { |
| /* feed idata into cache */ |
| item.inum = ni->mft_no; |
| item.ni = ni; |
| item.pathname = (const char*)NULL; |
| item.varsize = 0; |
| debug_cached_inode(ni); |
| ntfs_enter_cache(ni->vol->nidata_cache, |
| GENERIC(&item), idata_cache_compare); |
| } |
| } else { |
| /* cache not ready or system file, really close */ |
| res = ntfs_inode_real_close(ni); |
| } |
| } else |
| res = 0; |
| #else |
| res = ntfs_inode_real_close(ni); |
| #endif |
| return (res); |
| } |
| |
| /** |
| * ntfs_extent_inode_open - load an extent inode and attach it to its base |
| * @base_ni: base ntfs inode |
| * @mref: mft reference of the extent inode to load (in little endian) |
| * |
| * First check if the extent inode @mref is already attached to the base ntfs |
| * inode @base_ni, and if so, return a pointer to the attached extent inode. |
| * |
| * If the extent inode is not already attached to the base inode, allocate an |
| * ntfs_inode structure and initialize it for the given inode @mref. @mref |
| * specifies the inode number / mft record to read, including the sequence |
| * number, which can be 0 if no sequence number checking is to be performed. |
| * |
| * Then, allocate a buffer for the mft record, read the mft record from the |
| * volume @base_ni->vol, and attach it to the ntfs_inode structure (->mrec). |
| * The mft record is mst deprotected and sanity checked for validity and we |
| * abort if deprotection or checks fail. |
| * |
| * Finally attach the ntfs inode to its base inode @base_ni and return a |
| * pointer to the ntfs_inode structure on success or NULL on error, with errno |
| * set to the error code. |
| * |
| * Note, extent inodes are never closed directly. They are automatically |
| * disposed off by the closing of the base inode. |
| */ |
| ntfs_inode *ntfs_extent_inode_open(ntfs_inode *base_ni, const leMFT_REF mref) |
| { |
| u64 mft_no = MREF_LE(mref); |
| VCN extent_vcn; |
| runlist_element *rl; |
| ntfs_volume *vol; |
| ntfs_inode *ni = NULL; |
| ntfs_inode **extent_nis; |
| int i; |
| |
| if (!base_ni) { |
| errno = EINVAL; |
| ntfs_log_perror("%s", __FUNCTION__); |
| return NULL; |
| } |
| |
| ntfs_log_enter("Opening extent inode %lld (base mft record %lld).\n", |
| (unsigned long long)mft_no, |
| (unsigned long long)base_ni->mft_no); |
| |
| if (!base_ni->mft_no) { |
| /* |
| * When getting extents of MFT, we must be sure |
| * they are in the MFT part which has already |
| * been mapped, otherwise we fall into an endless |
| * recursion. |
| * Situations have been met where extents locations |
| * are described in themselves. |
| * This is a severe error which chkdsk cannot fix. |
| */ |
| vol = base_ni->vol; |
| extent_vcn = mft_no << vol->mft_record_size_bits |
| >> vol->cluster_size_bits; |
| rl = vol->mft_na->rl; |
| if (rl) { |
| while (rl->length |
| && ((rl->vcn + rl->length) <= extent_vcn)) |
| rl++; |
| } |
| if (!rl || (rl->lcn < 0)) { |
| ntfs_log_error("MFT is corrupt, cannot read" |
| " its unmapped extent record %lld\n", |
| (long long)mft_no); |
| ntfs_log_error("Note : chkdsk cannot fix this," |
| " try ntfsfix\n"); |
| errno = EIO; |
| ni = (ntfs_inode*)NULL; |
| goto out; |
| } |
| } |
| |
| /* Is the extent inode already open and attached to the base inode? */ |
| if (base_ni->nr_extents > 0) { |
| extent_nis = base_ni->extent_nis; |
| for (i = 0; i < base_ni->nr_extents; i++) { |
| u16 seq_no; |
| |
| ni = extent_nis[i]; |
| if (mft_no != ni->mft_no) |
| continue; |
| /* Verify the sequence number if given. */ |
| seq_no = MSEQNO_LE(mref); |
| if (seq_no && seq_no != le16_to_cpu( |
| ni->mrec->sequence_number)) { |
| errno = EIO; |
| ntfs_log_perror("Found stale extent mft " |
| "reference mft=%lld", |
| (long long)ni->mft_no); |
| goto out; |
| } |
| goto out; |
| } |
| } |
| /* Wasn't there, we need to load the extent inode. */ |
| ni = __ntfs_inode_allocate(base_ni->vol); |
| if (!ni) |
| goto out; |
| if (ntfs_file_record_read(base_ni->vol, le64_to_cpu(mref), &ni->mrec, NULL)) |
| goto err_out; |
| ni->mft_no = mft_no; |
| ni->nr_extents = -1; |
| ni->base_ni = base_ni; |
| /* Attach extent inode to base inode, reallocating memory if needed. */ |
| if (!(base_ni->nr_extents & 3)) { |
| i = (base_ni->nr_extents + 4) * sizeof(ntfs_inode *); |
| |
| extent_nis = ntfs_malloc(i); |
| if (!extent_nis) |
| goto err_out; |
| if (base_ni->nr_extents) { |
| memcpy(extent_nis, base_ni->extent_nis, |
| i - 4 * sizeof(ntfs_inode *)); |
| free(base_ni->extent_nis); |
| } |
| base_ni->extent_nis = extent_nis; |
| } |
| base_ni->extent_nis[base_ni->nr_extents++] = ni; |
| out: |
| ntfs_log_leave("\n"); |
| return ni; |
| err_out: |
| __ntfs_inode_release(ni); |
| ni = NULL; |
| goto out; |
| } |
| |
| /** |
| * ntfs_inode_attach_all_extents - attach all extents for target inode |
| * @ni: opened ntfs inode for which perform attach |
| * |
| * Return 0 on success and -1 on error with errno set to the error code. |
| */ |
| int ntfs_inode_attach_all_extents(ntfs_inode *ni) |
| { |
| ATTR_LIST_ENTRY *ale; |
| u64 prev_attached = 0; |
| |
| if (!ni) { |
| ntfs_log_trace("Invalid arguments.\n"); |
| errno = EINVAL; |
| return -1; |
| } |
| |
| if (ni->nr_extents == -1) |
| ni = ni->base_ni; |
| |
| ntfs_log_trace("Entering for inode 0x%llx.\n", (long long) ni->mft_no); |
| |
| /* Inode haven't got attribute list, thus nothing to attach. */ |
| if (!NInoAttrList(ni)) |
| return 0; |
| |
| if (!ni->attr_list) { |
| ntfs_log_trace("Corrupt in-memory struct.\n"); |
| errno = EINVAL; |
| return -1; |
| } |
| |
| /* Walk through attribute list and attach all extents. */ |
| errno = 0; |
| ale = (ATTR_LIST_ENTRY *)ni->attr_list; |
| while ((u8*)ale < ni->attr_list + ni->attr_list_size) { |
| if (ni->mft_no != MREF_LE(ale->mft_reference) && |
| prev_attached != MREF_LE(ale->mft_reference)) { |
| if (!ntfs_extent_inode_open(ni, ale->mft_reference)) { |
| ntfs_log_trace("Couldn't attach extent inode.\n"); |
| return -1; |
| } |
| prev_attached = MREF_LE(ale->mft_reference); |
| } |
| ale = (ATTR_LIST_ENTRY *)((u8*)ale + le16_to_cpu(ale->length)); |
| } |
| return 0; |
| } |
| |
| /** |
| * ntfs_inode_sync_standard_information - update standard information attribute |
| * @ni: ntfs inode to update standard information |
| * |
| * Return 0 on success or -1 on error with errno set to the error code. |
| */ |
| static int ntfs_inode_sync_standard_information(ntfs_inode *ni) |
| { |
| ntfs_attr_search_ctx *ctx; |
| STANDARD_INFORMATION *std_info; |
| u32 lth; |
| le32 lthle; |
| |
| ntfs_log_trace("Entering for inode %lld\n", (long long)ni->mft_no); |
| |
| ctx = ntfs_attr_get_search_ctx(ni, NULL); |
| if (!ctx) |
| return -1; |
| if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, AT_UNNAMED, |
| 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { |
| ntfs_log_perror("Failed to sync standard info (inode %lld)", |
| (long long)ni->mft_no); |
| ntfs_attr_put_search_ctx(ctx); |
| return -1; |
| } |
| std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->value_offset)); |
| std_info->file_attributes = ni->flags; |
| if (!test_nino_flag(ni, TimesSet)) { |
| std_info->creation_time = ni->creation_time; |
| std_info->last_data_change_time = ni->last_data_change_time; |
| std_info->last_mft_change_time = ni->last_mft_change_time; |
| std_info->last_access_time = ni->last_access_time; |
| } |
| |
| /* JPA update v3.x extensions, ensuring consistency */ |
| |
| lthle = ctx->attr->length; |
| lth = le32_to_cpu(lthle); |
| if (test_nino_flag(ni, v3_Extensions) |
| && (lth <= sizeof(STANDARD_INFORMATION))) |
| ntfs_log_error("bad sync of standard information\n"); |
| |
| if (lth > sizeof(STANDARD_INFORMATION)) { |
| std_info->owner_id = ni->owner_id; |
| std_info->security_id = ni->security_id; |
| std_info->quota_charged = ni->quota_charged; |
| std_info->usn = ni->usn; |
| } |
| ntfs_inode_mark_dirty(ctx->ntfs_ino); |
| ntfs_attr_put_search_ctx(ctx); |
| return 0; |
| } |
| |
| /** |
| * ntfs_inode_sync_file_name - update FILE_NAME attributes |
| * @ni: ntfs inode to update FILE_NAME attributes |
| * |
| * Update all FILE_NAME attributes for inode @ni in the index. |
| * |
| * Return 0 on success or -1 on error with errno set to the error code. |
| */ |
| static int ntfs_inode_sync_file_name(ntfs_inode *ni, ntfs_inode *dir_ni) |
| { |
| ntfs_attr_search_ctx *ctx = NULL; |
| ntfs_index_context *ictx; |
| ntfs_inode *index_ni; |
| FILE_NAME_ATTR *fn; |
| FILE_NAME_ATTR *fnx; |
| REPARSE_POINT *rpp; |
| le32 reparse_tag; |
| int err = 0; |
| |
| ntfs_log_trace("Entering for inode %lld\n", (long long)ni->mft_no); |
| |
| ctx = ntfs_attr_get_search_ctx(ni, NULL); |
| if (!ctx) { |
| err = errno; |
| goto err_out; |
| } |
| /* Collect the reparse tag, if any */ |
| reparse_tag = const_cpu_to_le32(0); |
| if (ni->flags & FILE_ATTR_REPARSE_POINT) { |
| if (!ntfs_attr_lookup(AT_REPARSE_POINT, NULL, |
| 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { |
| rpp = (REPARSE_POINT*)((u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->value_offset)); |
| reparse_tag = rpp->reparse_tag; |
| } |
| ntfs_attr_reinit_search_ctx(ctx); |
| } |
| /* Walk through all FILE_NAME attributes and update them. */ |
| while (!ntfs_attr_lookup(AT_FILE_NAME, NULL, 0, 0, 0, NULL, 0, ctx)) { |
| fn = (FILE_NAME_ATTR *)((u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->value_offset)); |
| if (MREF_LE(fn->parent_directory) == ni->mft_no) { |
| /* |
| * WARNING: We cheat here and obtain 2 attribute |
| * search contexts for one inode (first we obtained |
| * above, second will be obtained inside |
| * ntfs_index_lookup), it's acceptable for library, |
| * but will deadlock in the kernel. |
| */ |
| index_ni = ni; |
| } else |
| if (dir_ni) |
| index_ni = dir_ni; |
| else |
| index_ni = ntfs_inode_open(ni->vol, |
| le64_to_cpu(fn->parent_directory)); |
| if (!index_ni) { |
| if (!err) |
| err = errno; |
| ntfs_log_perror("Failed to open inode %lld with index", |
| (long long)le64_to_cpu(fn->parent_directory)); |
| continue; |
| } |
| ictx = ntfs_index_ctx_get(index_ni, NTFS_INDEX_I30, 4); |
| if (!ictx) { |
| if (!err) |
| err = errno; |
| ntfs_log_perror("Failed to get index ctx, inode %lld", |
| (long long)index_ni->mft_no); |
| if ((ni != index_ni) && !dir_ni |
| && ntfs_inode_close(index_ni) && !err) |
| err = errno; |
| continue; |
| } |
| if (ntfs_index_lookup(fn, sizeof(FILE_NAME_ATTR), ictx)) { |
| if (!err) { |
| if (errno == ENOENT) |
| err = EIO; |
| else |
| err = errno; |
| } |
| ntfs_log_perror("Index lookup failed, inode %lld", |
| (long long)index_ni->mft_no); |
| ntfs_index_ctx_put(ictx); |
| if (ni != index_ni && ntfs_inode_close(index_ni) && !err) |
| err = errno; |
| continue; |
| } |
| /* Update flags and file size. */ |
| fnx = (FILE_NAME_ATTR *)ictx->data; |
| fnx->file_attributes = |
| (fnx->file_attributes & ~FILE_ATTR_VALID_FLAGS) | |
| (ni->flags & FILE_ATTR_VALID_FLAGS); |
| if (ni->mrec->flags & MFT_RECORD_IS_DIRECTORY) |
| fnx->data_size = fnx->allocated_size |
| = const_cpu_to_sle64(0); |
| else { |
| fnx->allocated_size = cpu_to_sle64(ni->allocated_size); |
| fnx->data_size = cpu_to_sle64(ni->data_size); |
| /* |
| * The file name record has also to be fixed if some |
| * attribute update implied the unnamed data to be |
| * made non-resident |
| */ |
| fn->allocated_size = fnx->allocated_size; |
| } |
| /* update or clear the reparse tag in the index */ |
| fnx->reparse_point_tag = reparse_tag; |
| if (!test_nino_flag(ni, TimesSet)) { |
| fnx->creation_time = ni->creation_time; |
| fnx->last_data_change_time = ni->last_data_change_time; |
| fnx->last_mft_change_time = ni->last_mft_change_time; |
| fnx->last_access_time = ni->last_access_time; |
| } else { |
| fnx->creation_time = fn->creation_time; |
| fnx->last_data_change_time = fn->last_data_change_time; |
| fnx->last_mft_change_time = fn->last_mft_change_time; |
| fnx->last_access_time = fn->last_access_time; |
| } |
| ntfs_index_entry_mark_dirty(ictx); |
| ntfs_index_ctx_put(ictx); |
| if ((ni != index_ni) && !dir_ni |
| && ntfs_inode_close(index_ni) && !err) |
| err = errno; |
| } |
| /* Check for real error occurred. */ |
| if (errno != ENOENT) { |
| err = errno; |
| ntfs_log_perror("Attribute lookup failed, inode %lld", |
| (long long)ni->mft_no); |
| goto err_out; |
| } |
| ntfs_attr_put_search_ctx(ctx); |
| if (err) { |
| errno = err; |
| return -1; |
| } |
| return 0; |
| err_out: |
| if (ctx) |
| ntfs_attr_put_search_ctx(ctx); |
| errno = err; |
| return -1; |
| } |
| |
| /** |
| * ntfs_inode_sync - write the inode (and its dirty extents) to disk |
| * @ni: ntfs inode to write |
| * |
| * Write the inode @ni to disk as well as its dirty extent inodes if such |
| * exist and @ni is a base inode. If @ni is an extent inode, only @ni is |
| * written completely disregarding its base inode and any other extent inodes. |
| * |
| * For a base inode with dirty extent inodes if any writes fail for whatever |
| * reason, the failing inode is skipped and the sync process is continued. At |
| * the end the error condition that brought about the failure is returned. Thus |
| * the smallest amount of data loss possible occurs. |
| * |
| * Return 0 on success or -1 on error with errno set to the error code. |
| * The following error codes are defined: |
| * EINVAL - Invalid arguments were passed to the function. |
| * EBUSY - Inode and/or one of its extents is busy, try again later. |
| * EIO - I/O error while writing the inode (or one of its extents). |
| */ |
| static int ntfs_inode_sync_in_dir(ntfs_inode *ni, ntfs_inode *dir_ni) |
| { |
| int ret = 0; |
| int err = 0; |
| if (!ni) { |
| errno = EINVAL; |
| ntfs_log_error("Failed to sync NULL inode\n"); |
| return -1; |
| } |
| |
| ntfs_log_enter("Entering for inode %lld\n", (long long)ni->mft_no); |
| |
| /* Update STANDARD_INFORMATION. */ |
| if ((ni->mrec->flags & MFT_RECORD_IN_USE) && ni->nr_extents != -1 && |
| ntfs_inode_sync_standard_information(ni)) { |
| if (!err || errno == EIO) { |
| err = errno; |
| if (err != EIO) |
| err = EBUSY; |
| } |
| } |
| |
| /* Update FILE_NAME's in the index. */ |
| if ((ni->mrec->flags & MFT_RECORD_IN_USE) && ni->nr_extents != -1 && |
| NInoFileNameTestAndClearDirty(ni) && |
| ntfs_inode_sync_file_name(ni, dir_ni)) { |
| if (!err || errno == EIO) { |
| err = errno; |
| if (err != EIO) |
| err = EBUSY; |
| } |
| ntfs_log_perror("Failed to sync FILE_NAME (inode %lld)", |
| (long long)ni->mft_no); |
| NInoFileNameSetDirty(ni); |
| } |
| |
| /* Write out attribute list from cache to disk. */ |
| if ((ni->mrec->flags & MFT_RECORD_IN_USE) && ni->nr_extents != -1 && |
| NInoAttrList(ni) && NInoAttrListTestAndClearDirty(ni)) { |
| ntfs_attr *na; |
| |
| na = ntfs_attr_open(ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0); |
| if (!na) { |
| if (!err || errno == EIO) { |
| err = errno; |
| if (err != EIO) |
| err = EBUSY; |
| ntfs_log_perror("Attribute list sync failed " |
| "(open, inode %lld)", |
| (long long)ni->mft_no); |
| } |
| NInoAttrListSetDirty(ni); |
| goto sync_inode; |
| } |
| |
| if (na->data_size == ni->attr_list_size) { |
| if (ntfs_attr_pwrite(na, 0, ni->attr_list_size, |
| ni->attr_list) != ni->attr_list_size) { |
| if (!err || errno == EIO) { |
| err = errno; |
| if (err != EIO) |
| err = EBUSY; |
| ntfs_log_perror("Attribute list sync " |
| "failed (write, inode %lld)", |
| (long long)ni->mft_no); |
| } |
| NInoAttrListSetDirty(ni); |
| } |
| } else { |
| err = EIO; |
| ntfs_log_error("Attribute list sync failed (bad size, " |
| "inode %lld)\n", (long long)ni->mft_no); |
| NInoAttrListSetDirty(ni); |
| } |
| ntfs_attr_close(na); |
| } |
| |
| sync_inode: |
| /* Write this inode out to the $MFT (and $MFTMirr if applicable). */ |
| if (NInoTestAndClearDirty(ni)) { |
| if (ntfs_mft_record_write(ni->vol, ni->mft_no, ni->mrec)) { |
| if (!err || errno == EIO) { |
| err = errno; |
| if (err != EIO) |
| err = EBUSY; |
| } |
| NInoSetDirty(ni); |
| ntfs_log_perror("MFT record sync failed, inode %lld", |
| (long long)ni->mft_no); |
| } |
| } |
| |
| /* If this is a base inode with extents write all dirty extents, too. */ |
| if (ni->nr_extents > 0) { |
| s32 i; |
| |
| for (i = 0; i < ni->nr_extents; ++i) { |
| ntfs_inode *eni; |
| |
| eni = ni->extent_nis[i]; |
| if (!NInoTestAndClearDirty(eni)) |
| continue; |
| |
| if (ntfs_mft_record_write(eni->vol, eni->mft_no, |
| eni->mrec)) { |
| if (!err || errno == EIO) { |
| err = errno; |
| if (err != EIO) |
| err = EBUSY; |
| } |
| NInoSetDirty(eni); |
| ntfs_log_perror("Extent MFT record sync failed," |
| " inode %lld/%lld", |
| (long long)ni->mft_no, |
| (long long)eni->mft_no); |
| } |
| } |
| } |
| |
| if (err) { |
| errno = err; |
| ret = -1; |
| } |
| |
| ntfs_log_leave("\n"); |
| return ret; |
| } |
| |
| int ntfs_inode_sync(ntfs_inode *ni) |
| { |
| return (ntfs_inode_sync_in_dir(ni, (ntfs_inode*)NULL)); |
| } |
| |
| /* |
| * Close an inode with an open parent inode |
| */ |
| |
| int ntfs_inode_close_in_dir(ntfs_inode *ni, ntfs_inode *dir_ni) |
| { |
| int res; |
| |
| res = ntfs_inode_sync_in_dir(ni, dir_ni); |
| if (res) { |
| if (errno != EIO) |
| errno = EBUSY; |
| } else |
| res = ntfs_inode_close(ni); |
| return (res); |
| } |
| |
| /** |
| * ntfs_inode_add_attrlist - add attribute list to inode and fill it |
| * @ni: opened ntfs inode to which add attribute list |
| * |
| * Return 0 on success or -1 on error with errno set to the error code. |
| * The following error codes are defined: |
| * EINVAL - Invalid arguments were passed to the function. |
| * EEXIST - Attribute list already exist. |
| * EIO - Input/Ouput error occurred. |
| * ENOMEM - Not enough memory to perform add. |
| */ |
| int ntfs_inode_add_attrlist(ntfs_inode *ni) |
| { |
| int err; |
| ntfs_attr_search_ctx *ctx; |
| u8 *al = NULL, *aln; |
| int al_len = 0; |
| ATTR_LIST_ENTRY *ale = NULL; |
| ntfs_attr *na; |
| |
| if (!ni) { |
| errno = EINVAL; |
| ntfs_log_perror("%s", __FUNCTION__); |
| return -1; |
| } |
| |
| ntfs_log_trace("inode %llu\n", (unsigned long long) ni->mft_no); |
| |
| if (NInoAttrList(ni) || ni->nr_extents) { |
| errno = EEXIST; |
| ntfs_log_perror("Inode already has attribute list"); |
| return -1; |
| } |
| |
| /* Form attribute list. */ |
| ctx = ntfs_attr_get_search_ctx(ni, NULL); |
| if (!ctx) { |
| err = errno; |
| goto err_out; |
| } |
| /* Walk through all attributes. */ |
| while (!ntfs_attr_lookup(AT_UNUSED, NULL, 0, 0, 0, NULL, 0, ctx)) { |
| |
| int ale_size; |
| |
| if (ctx->attr->type == AT_ATTRIBUTE_LIST) { |
| err = EIO; |
| ntfs_log_perror("Attribute list already present"); |
| goto put_err_out; |
| } |
| |
| ale_size = (sizeof(ATTR_LIST_ENTRY) + sizeof(ntfschar) * |
| ctx->attr->name_length + 7) & ~7; |
| al_len += ale_size; |
| |
| aln = realloc(al, al_len); |
| if (!aln) { |
| err = errno; |
| ntfs_log_perror("Failed to realloc %d bytes", al_len); |
| goto put_err_out; |
| } |
| ale = (ATTR_LIST_ENTRY *)(aln + ((u8 *)ale - al)); |
| al = aln; |
| |
| memset(ale, 0, ale_size); |
| |
| /* Add attribute to attribute list. */ |
| ale->type = ctx->attr->type; |
| ale->length = cpu_to_le16((sizeof(ATTR_LIST_ENTRY) + |
| sizeof(ntfschar) * ctx->attr->name_length + 7) & ~7); |
| ale->name_length = ctx->attr->name_length; |
| ale->name_offset = (u8 *)ale->name - (u8 *)ale; |
| if (ctx->attr->non_resident) |
| ale->lowest_vcn = ctx->attr->lowest_vcn; |
| else |
| ale->lowest_vcn = const_cpu_to_sle64(0); |
| ale->mft_reference = MK_LE_MREF(ni->mft_no, |
| le16_to_cpu(ni->mrec->sequence_number)); |
| ale->instance = ctx->attr->instance; |
| memcpy(ale->name, (u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->name_offset), |
| ctx->attr->name_length * sizeof(ntfschar)); |
| ale = (ATTR_LIST_ENTRY *)(al + al_len); |
| } |
| /* Check for real error occurred. */ |
| if (errno != ENOENT) { |
| err = errno; |
| ntfs_log_perror("%s: Attribute lookup failed, inode %lld", |
| __FUNCTION__, (long long)ni->mft_no); |
| goto put_err_out; |
| } |
| |
| /* Set in-memory attribute list. */ |
| ni->attr_list = al; |
| ni->attr_list_size = al_len; |
| NInoSetAttrList(ni); |
| NInoAttrListSetDirty(ni); |
| |
| /* Free space if there is not enough it for $ATTRIBUTE_LIST. */ |
| if (le32_to_cpu(ni->mrec->bytes_allocated) - |
| le32_to_cpu(ni->mrec->bytes_in_use) < |
| offsetof(ATTR_RECORD, resident_end)) { |
| if (ntfs_inode_free_space(ni, |
| offsetof(ATTR_RECORD, resident_end))) { |
| /* Failed to free space. */ |
| err = errno; |
| ntfs_log_perror("Failed to free space for attrlist"); |
| goto rollback; |
| } |
| } |
| |
| /* Add $ATTRIBUTE_LIST to mft record. */ |
| if (ntfs_resident_attr_record_add(ni, |
| AT_ATTRIBUTE_LIST, NULL, 0, NULL, 0, const_cpu_to_le16(0)) < 0) { |
| err = errno; |
| ntfs_log_perror("Couldn't add $ATTRIBUTE_LIST to MFT"); |
| goto rollback; |
| } |
| |
| /* Resize it. */ |
| na = ntfs_attr_open(ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0); |
| if (!na) { |
| err = errno; |
| ntfs_log_perror("Failed to open just added $ATTRIBUTE_LIST"); |
| goto remove_attrlist_record; |
| } |
| if (ntfs_attr_truncate(na, al_len)) { |
| err = errno; |
| ntfs_log_perror("Failed to resize just added $ATTRIBUTE_LIST"); |
| ntfs_attr_close(na); |
| goto remove_attrlist_record;; |
| } |
| |
| ntfs_attr_put_search_ctx(ctx); |
| ntfs_attr_close(na); |
| return 0; |
| |
| remove_attrlist_record: |
| /* Prevent ntfs_attr_recorm_rm from freeing attribute list. */ |
| ni->attr_list = NULL; |
| NInoClearAttrList(ni); |
| /* Remove $ATTRIBUTE_LIST record. */ |
| ntfs_attr_reinit_search_ctx(ctx); |
| if (!ntfs_attr_lookup(AT_ATTRIBUTE_LIST, NULL, 0, |
| CASE_SENSITIVE, 0, NULL, 0, ctx)) { |
| if (ntfs_attr_record_rm(ctx)) |
| ntfs_log_perror("Rollback failed to remove attrlist"); |
| } else |
| ntfs_log_perror("Rollback failed to find attrlist"); |
| /* Setup back in-memory runlist. */ |
| ni->attr_list = al; |
| ni->attr_list_size = al_len; |
| NInoSetAttrList(ni); |
| rollback: |
| /* |
| * Scan attribute list for attributes that placed not in the base MFT |
| * record and move them to it. |
| */ |
| ntfs_attr_reinit_search_ctx(ctx); |
| ale = (ATTR_LIST_ENTRY*)al; |
| while ((u8*)ale < al + al_len) { |
| if (MREF_LE(ale->mft_reference) != ni->mft_no) { |
| if (!ntfs_attr_lookup(ale->type, ale->name, |
| ale->name_length, |
| CASE_SENSITIVE, |
| sle64_to_cpu(ale->lowest_vcn), |
| NULL, 0, ctx)) { |
| if (ntfs_attr_record_move_to(ctx, ni)) |
| ntfs_log_perror("Rollback failed to " |
| "move attribute"); |
| } else |
| ntfs_log_perror("Rollback failed to find attr"); |
| ntfs_attr_reinit_search_ctx(ctx); |
| } |
| ale = (ATTR_LIST_ENTRY*)((u8*)ale + le16_to_cpu(ale->length)); |
| } |
| /* Remove in-memory attribute list. */ |
| ni->attr_list = NULL; |
| ni->attr_list_size = 0; |
| NInoClearAttrList(ni); |
| NInoAttrListClearDirty(ni); |
| put_err_out: |
| ntfs_attr_put_search_ctx(ctx); |
| err_out: |
| free(al); |
| errno = err; |
| return -1; |
| } |
| |
| /** |
| * ntfs_inode_free_space - free space in the MFT record of an inode |
| * @ni: ntfs inode in which MFT record needs more free space |
| * @size: amount of space needed to free |
| * |
| * Return 0 on success or -1 on error with errno set to the error code. |
| */ |
| int ntfs_inode_free_space(ntfs_inode *ni, int size) |
| { |
| ntfs_attr_search_ctx *ctx; |
| int freed; |
| |
| if (!ni || size < 0) { |
| errno = EINVAL; |
| ntfs_log_perror("%s: ni=%p size=%d", __FUNCTION__, ni, size); |
| return -1; |
| } |
| |
| ntfs_log_trace("Entering for inode %lld, size %d\n", |
| (unsigned long long)ni->mft_no, size); |
| |
| freed = (le32_to_cpu(ni->mrec->bytes_allocated) - |
| le32_to_cpu(ni->mrec->bytes_in_use)); |
| |
| if (size <= freed) |
| return 0; |
| |
| ctx = ntfs_attr_get_search_ctx(ni, NULL); |
| if (!ctx) |
| return -1; |
| /* |
| * $STANDARD_INFORMATION and $ATTRIBUTE_LIST must stay in the base MFT |
| * record, so position search context on the first attribute after them. |
| */ |
| if (ntfs_attr_position(AT_FILE_NAME, ctx)) |
| goto put_err_out; |
| |
| while (1) { |
| int record_size; |
| /* |
| * Check whether attribute is from different MFT record. If so, |
| * find next, because we don't need such. |
| */ |
| while (ctx->ntfs_ino->mft_no != ni->mft_no) { |
| retry: |
| if (ntfs_attr_position(AT_UNUSED, ctx)) |
| goto put_err_out; |
| } |
| |
| if (ntfs_inode_base(ctx->ntfs_ino)->mft_no == FILE_MFT && |
| ctx->attr->type == AT_DATA) |
| goto retry; |
| |
| if (ctx->attr->type == AT_INDEX_ROOT) |
| goto retry; |
| |
| record_size = le32_to_cpu(ctx->attr->length); |
| |
| if (ntfs_attr_record_move_away(ctx, 0)) { |
| ntfs_log_perror("Failed to move out attribute #2"); |
| break; |
| } |
| freed += record_size; |
| |
| /* Check whether we are done. */ |
| if (size <= freed) { |
| ntfs_attr_put_search_ctx(ctx); |
| return 0; |
| } |
| /* |
| * Reposition to first attribute after $STANDARD_INFORMATION |
| * and $ATTRIBUTE_LIST instead of simply skipping this attribute |
| * because in the case when we have got only in-memory attribute |
| * list then ntfs_attr_lookup will fail when it tries to find |
| * $ATTRIBUTE_LIST. |
| */ |
| ntfs_attr_reinit_search_ctx(ctx); |
| if (ntfs_attr_position(AT_FILE_NAME, ctx)) |
| break; |
| } |
| put_err_out: |
| ntfs_attr_put_search_ctx(ctx); |
| if (errno == ENOSPC) |
| ntfs_log_trace("No attributes left that could be moved out.\n"); |
| return -1; |
| } |
| |
| /** |
| * ntfs_inode_update_times - update selected time fields for ntfs inode |
| * @ni: ntfs inode for which update time fields |
| * @mask: select which time fields should be updated |
| * |
| * This function updates time fields to current time. Fields to update are |
| * selected using @mask (see enum @ntfs_time_update_flags for posssible values). |
| */ |
| void ntfs_inode_update_times(ntfs_inode *ni, ntfs_time_update_flags mask) |
| { |
| ntfs_time now; |
| |
| if (!ni) { |
| ntfs_log_error("%s(): Invalid arguments.\n", __FUNCTION__); |
| return; |
| } |
| |
| if ((ni->mft_no < FILE_first_user && ni->mft_no != FILE_root) || |
| NVolReadOnly(ni->vol) || !mask) |
| return; |
| |
| now = ntfs_current_time(); |
| if (mask & NTFS_UPDATE_ATIME) |
| ni->last_access_time = now; |
| if (mask & NTFS_UPDATE_MTIME) |
| ni->last_data_change_time = now; |
| if (mask & NTFS_UPDATE_CTIME) |
| ni->last_mft_change_time = now; |
| |
| NInoFileNameSetDirty(ni); |
| NInoSetDirty(ni); |
| } |
| |
| /** |
| * ntfs_inode_badclus_bad - check for $Badclus:$Bad data attribute |
| * @mft_no: mft record number where @attr is present |
| * @attr: attribute record used to check for the $Bad attribute |
| * |
| * Check if the mft record given by @mft_no and @attr contains the bad sector |
| * list. Please note that mft record numbers describing $Badclus extent inodes |
| * will not match the current $Badclus:$Bad check. |
| * |
| * On success return 1 if the file is $Badclus:$Bad, otherwise return 0. |
| * On error return -1 with errno set to the error code. |
| */ |
| int ntfs_inode_badclus_bad(u64 mft_no, ATTR_RECORD *attr) |
| { |
| int len, ret = 0; |
| ntfschar *ustr; |
| |
| if (!attr) { |
| ntfs_log_error("Invalid argument.\n"); |
| errno = EINVAL; |
| return -1; |
| } |
| |
| if (mft_no != FILE_BadClus) |
| return 0; |
| |
| if (attr->type != AT_DATA) |
| return 0; |
| |
| if ((ustr = ntfs_str2ucs("$Bad", &len)) == NULL) { |
| ntfs_log_perror("Couldn't convert '$Bad' to Unicode"); |
| return -1; |
| } |
| |
| if (ustr && ntfs_names_are_equal(ustr, len, |
| (ntfschar *)((u8 *)attr + le16_to_cpu(attr->name_offset)), |
| attr->name_length, 0, NULL, 0)) |
| ret = 1; |
| |
| ntfs_ucsfree(ustr); |
| |
| return ret; |
| } |
| |
| #ifdef HAVE_SETXATTR /* extended attributes interface required */ |
| |
| /* |
| * Get high precision NTFS times |
| * |
| * They are returned in following order : create, update, access, change |
| * provided they fit in requested size. |
| * |
| * Returns the modified size if successfull (or 32 if buffer size is null) |
| * -errno if failed |
| */ |
| |
| int ntfs_inode_get_times(ntfs_inode *ni, char *value, size_t size) |
| { |
| ntfs_attr_search_ctx *ctx; |
| STANDARD_INFORMATION *std_info; |
| u64 *times; |
| int ret; |
| |
| ret = 0; |
| ctx = ntfs_attr_get_search_ctx(ni, NULL); |
| if (ctx) { |
| if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, AT_UNNAMED, |
| 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { |
| ntfs_log_perror("Failed to get standard info (inode %lld)", |
| (long long)ni->mft_no); |
| } else { |
| std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->value_offset)); |
| if (value && (size >= 8)) { |
| times = (u64*)value; |
| times[0] = sle64_to_cpu(std_info->creation_time); |
| ret = 8; |
| if (size >= 16) { |
| times[1] = sle64_to_cpu(std_info->last_data_change_time); |
| ret = 16; |
| } |
| if (size >= 24) { |
| times[2] = sle64_to_cpu(std_info->last_access_time); |
| ret = 24; |
| } |
| if (size >= 32) { |
| times[3] = sle64_to_cpu(std_info->last_mft_change_time); |
| ret = 32; |
| } |
| } else |
| if (!size) |
| ret = 32; |
| else |
| ret = -ERANGE; |
| } |
| ntfs_attr_put_search_ctx(ctx); |
| } |
| return (ret ? ret : -errno); |
| } |
| |
| /* |
| * Set high precision NTFS times |
| * |
| * They are expected in this order : create, update, access |
| * provided they are present in input. The change time is set to |
| * current time. |
| * |
| * The times are inserted directly in the standard_information and |
| * file names attributes to avoid manipulating low precision times |
| * |
| * Returns 0 if success |
| * -1 if there were an error (described by errno) |
| */ |
| |
| int ntfs_inode_set_times(ntfs_inode *ni, const char *value, size_t size, |
| int flags) |
| { |
| ntfs_attr_search_ctx *ctx; |
| STANDARD_INFORMATION *std_info; |
| FILE_NAME_ATTR *fn; |
| const u64 *times; |
| ntfs_time now; |
| int cnt; |
| int ret; |
| |
| ret = -1; |
| if ((size >= 8) && !(flags & XATTR_CREATE)) { |
| times = (const u64*)value; |
| now = ntfs_current_time(); |
| /* update the standard information attribute */ |
| ctx = ntfs_attr_get_search_ctx(ni, NULL); |
| if (ctx) { |
| if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, |
| AT_UNNAMED, 0, CASE_SENSITIVE, |
| 0, NULL, 0, ctx)) { |
| ntfs_log_perror("Failed to get standard info (inode %lld)", |
| (long long)ni->mft_no); |
| } else { |
| std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->value_offset)); |
| /* |
| * Mark times set to avoid overwriting |
| * them when the inode is closed. |
| * The inode structure must also be updated |
| * (with loss of precision) because of cacheing. |
| * TODO : use NTFS precision in inode, and |
| * return sub-second times in getattr() |
| */ |
| set_nino_flag(ni, TimesSet); |
| std_info->creation_time = cpu_to_sle64(times[0]); |
| ni->creation_time |
| = std_info->creation_time; |
| if (size >= 16) { |
| std_info->last_data_change_time = cpu_to_sle64(times[1]); |
| ni->last_data_change_time |
| = std_info->last_data_change_time; |
| } |
| if (size >= 24) { |
| std_info->last_access_time = cpu_to_sle64(times[2]); |
| ni->last_access_time |
| = std_info->last_access_time; |
| } |
| std_info->last_mft_change_time = now; |
| ni->last_mft_change_time = now; |
| ntfs_inode_mark_dirty(ctx->ntfs_ino); |
| NInoFileNameSetDirty(ni); |
| |
| /* update the file names attributes */ |
| ntfs_attr_reinit_search_ctx(ctx); |
| cnt = 0; |
| while (!ntfs_attr_lookup(AT_FILE_NAME, |
| AT_UNNAMED, 0, CASE_SENSITIVE, |
| 0, NULL, 0, ctx)) { |
| fn = (FILE_NAME_ATTR*)((u8 *)ctx->attr + |
| le16_to_cpu(ctx->attr->value_offset)); |
| fn->creation_time |
| = cpu_to_sle64(times[0]); |
| if (size >= 16) |
| fn->last_data_change_time |
| = cpu_to_sle64(times[1]); |
| if (size >= 24) |
| fn->last_access_time |
| = cpu_to_sle64(times[2]); |
| fn->last_mft_change_time = now; |
| cnt++; |
| } |
| if (cnt) |
| ret = 0; |
| else { |
| ntfs_log_perror("Failed to get file names (inode %lld)", |
| (long long)ni->mft_no); |
| } |
| } |
| ntfs_attr_put_search_ctx(ctx); |
| } |
| } else |
| if (size < 8) |
| errno = ERANGE; |
| else |
| errno = EEXIST; |
| return (ret); |
| } |
| |
| #endif /* HAVE_SETXATTR */ |