1987da915Sopenharmony_ci/** 2987da915Sopenharmony_ci * inode.c - Inode handling code. Originated from the Linux-NTFS project. 3987da915Sopenharmony_ci * 4987da915Sopenharmony_ci * Copyright (c) 2002-2005 Anton Altaparmakov 5987da915Sopenharmony_ci * Copyright (c) 2002-2008 Szabolcs Szakacsits 6987da915Sopenharmony_ci * Copyright (c) 2004-2007 Yura Pakhuchiy 7987da915Sopenharmony_ci * Copyright (c) 2004-2005 Richard Russon 8987da915Sopenharmony_ci * Copyright (c) 2009-2010 Jean-Pierre Andre 9987da915Sopenharmony_ci * 10987da915Sopenharmony_ci * This program/include file is free software; you can redistribute it and/or 11987da915Sopenharmony_ci * modify it under the terms of the GNU General Public License as published 12987da915Sopenharmony_ci * by the Free Software Foundation; either version 2 of the License, or 13987da915Sopenharmony_ci * (at your option) any later version. 14987da915Sopenharmony_ci * 15987da915Sopenharmony_ci * This program/include file is distributed in the hope that it will be 16987da915Sopenharmony_ci * useful, but WITHOUT ANY WARRANTY; without even the implied warranty 17987da915Sopenharmony_ci * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 18987da915Sopenharmony_ci * GNU General Public License for more details. 19987da915Sopenharmony_ci * 20987da915Sopenharmony_ci * You should have received a copy of the GNU General Public License 21987da915Sopenharmony_ci * along with this program (in the main directory of the NTFS-3G 22987da915Sopenharmony_ci * distribution in the file COPYING); if not, write to the Free Software 23987da915Sopenharmony_ci * Foundation,Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 24987da915Sopenharmony_ci */ 25987da915Sopenharmony_ci 26987da915Sopenharmony_ci#ifdef HAVE_CONFIG_H 27987da915Sopenharmony_ci#include "config.h" 28987da915Sopenharmony_ci#endif 29987da915Sopenharmony_ci 30987da915Sopenharmony_ci#ifdef HAVE_STDLIB_H 31987da915Sopenharmony_ci#include <stdlib.h> 32987da915Sopenharmony_ci#endif 33987da915Sopenharmony_ci#ifdef HAVE_STRING_H 34987da915Sopenharmony_ci#include <string.h> 35987da915Sopenharmony_ci#endif 36987da915Sopenharmony_ci#ifdef HAVE_ERRNO_H 37987da915Sopenharmony_ci#include <errno.h> 38987da915Sopenharmony_ci#endif 39987da915Sopenharmony_ci 40987da915Sopenharmony_ci#include "param.h" 41987da915Sopenharmony_ci#include "compat.h" 42987da915Sopenharmony_ci#include "types.h" 43987da915Sopenharmony_ci#include "volume.h" 44987da915Sopenharmony_ci#include "cache.h" 45987da915Sopenharmony_ci#include "inode.h" 46987da915Sopenharmony_ci#include "attrib.h" 47987da915Sopenharmony_ci#include "debug.h" 48987da915Sopenharmony_ci#include "mft.h" 49987da915Sopenharmony_ci#include "attrlist.h" 50987da915Sopenharmony_ci#include "runlist.h" 51987da915Sopenharmony_ci#include "lcnalloc.h" 52987da915Sopenharmony_ci#include "index.h" 53987da915Sopenharmony_ci#include "dir.h" 54987da915Sopenharmony_ci#include "ntfstime.h" 55987da915Sopenharmony_ci#include "logging.h" 56987da915Sopenharmony_ci#include "misc.h" 57987da915Sopenharmony_ci#include "xattrs.h" 58987da915Sopenharmony_ci 59987da915Sopenharmony_cintfs_inode *ntfs_inode_base(ntfs_inode *ni) 60987da915Sopenharmony_ci{ 61987da915Sopenharmony_ci if (ni->nr_extents == -1) 62987da915Sopenharmony_ci return ni->base_ni; 63987da915Sopenharmony_ci return ni; 64987da915Sopenharmony_ci} 65987da915Sopenharmony_ci 66987da915Sopenharmony_ci/** 67987da915Sopenharmony_ci * ntfs_inode_mark_dirty - set the inode (and its base inode if it exists) dirty 68987da915Sopenharmony_ci * @ni: ntfs inode to set dirty 69987da915Sopenharmony_ci * 70987da915Sopenharmony_ci * Set the inode @ni dirty so it is written out later (at the latest at 71987da915Sopenharmony_ci * ntfs_inode_close() time). If @ni is an extent inode, set the base inode 72987da915Sopenharmony_ci * dirty, too. 73987da915Sopenharmony_ci * 74987da915Sopenharmony_ci * This function cannot fail. 75987da915Sopenharmony_ci */ 76987da915Sopenharmony_civoid ntfs_inode_mark_dirty(ntfs_inode *ni) 77987da915Sopenharmony_ci{ 78987da915Sopenharmony_ci NInoSetDirty(ni); 79987da915Sopenharmony_ci if (ni->nr_extents == -1) 80987da915Sopenharmony_ci NInoSetDirty(ni->base_ni); 81987da915Sopenharmony_ci} 82987da915Sopenharmony_ci 83987da915Sopenharmony_ci/** 84987da915Sopenharmony_ci * __ntfs_inode_allocate - Create and initialise an NTFS inode object 85987da915Sopenharmony_ci * @vol: 86987da915Sopenharmony_ci * 87987da915Sopenharmony_ci * Description... 88987da915Sopenharmony_ci * 89987da915Sopenharmony_ci * Returns: 90987da915Sopenharmony_ci */ 91987da915Sopenharmony_cistatic ntfs_inode *__ntfs_inode_allocate(ntfs_volume *vol) 92987da915Sopenharmony_ci{ 93987da915Sopenharmony_ci ntfs_inode *ni; 94987da915Sopenharmony_ci 95987da915Sopenharmony_ci ni = (ntfs_inode*)ntfs_calloc(sizeof(ntfs_inode)); 96987da915Sopenharmony_ci if (ni) 97987da915Sopenharmony_ci ni->vol = vol; 98987da915Sopenharmony_ci return ni; 99987da915Sopenharmony_ci} 100987da915Sopenharmony_ci 101987da915Sopenharmony_ci/** 102987da915Sopenharmony_ci * ntfs_inode_allocate - Create an NTFS inode object 103987da915Sopenharmony_ci * @vol: 104987da915Sopenharmony_ci * 105987da915Sopenharmony_ci * Description... 106987da915Sopenharmony_ci * 107987da915Sopenharmony_ci * Returns: 108987da915Sopenharmony_ci */ 109987da915Sopenharmony_cintfs_inode *ntfs_inode_allocate(ntfs_volume *vol) 110987da915Sopenharmony_ci{ 111987da915Sopenharmony_ci return __ntfs_inode_allocate(vol); 112987da915Sopenharmony_ci} 113987da915Sopenharmony_ci 114987da915Sopenharmony_ci/** 115987da915Sopenharmony_ci * __ntfs_inode_release - Destroy an NTFS inode object 116987da915Sopenharmony_ci * @ni: 117987da915Sopenharmony_ci * 118987da915Sopenharmony_ci * Description... 119987da915Sopenharmony_ci * 120987da915Sopenharmony_ci * Returns: 121987da915Sopenharmony_ci */ 122987da915Sopenharmony_cistatic void __ntfs_inode_release(ntfs_inode *ni) 123987da915Sopenharmony_ci{ 124987da915Sopenharmony_ci if (NInoDirty(ni)) 125987da915Sopenharmony_ci ntfs_log_error("Releasing dirty inode %lld!\n", 126987da915Sopenharmony_ci (long long)ni->mft_no); 127987da915Sopenharmony_ci if (NInoAttrList(ni) && ni->attr_list) 128987da915Sopenharmony_ci free(ni->attr_list); 129987da915Sopenharmony_ci free(ni->mrec); 130987da915Sopenharmony_ci free(ni); 131987da915Sopenharmony_ci return; 132987da915Sopenharmony_ci} 133987da915Sopenharmony_ci 134987da915Sopenharmony_ci/** 135987da915Sopenharmony_ci * ntfs_inode_open - open an inode ready for access 136987da915Sopenharmony_ci * @vol: volume to get the inode from 137987da915Sopenharmony_ci * @mref: inode number / mft record number to open 138987da915Sopenharmony_ci * 139987da915Sopenharmony_ci * Allocate an ntfs_inode structure and initialize it for the given inode 140987da915Sopenharmony_ci * specified by @mref. @mref specifies the inode number / mft record to read, 141987da915Sopenharmony_ci * including the sequence number, which can be 0 if no sequence number checking 142987da915Sopenharmony_ci * is to be performed. 143987da915Sopenharmony_ci * 144987da915Sopenharmony_ci * Then, allocate a buffer for the mft record, read the mft record from the 145987da915Sopenharmony_ci * volume @vol, and attach it to the ntfs_inode structure (->mrec). The 146987da915Sopenharmony_ci * mft record is mst deprotected and sanity checked for validity and we abort 147987da915Sopenharmony_ci * if deprotection or checks fail. 148987da915Sopenharmony_ci * 149987da915Sopenharmony_ci * Finally, search for an attribute list attribute in the mft record and if one 150987da915Sopenharmony_ci * is found, load the attribute list attribute value and attach it to the 151987da915Sopenharmony_ci * ntfs_inode structure (->attr_list). Also set the NI_AttrList bit to indicate 152987da915Sopenharmony_ci * this. 153987da915Sopenharmony_ci * 154987da915Sopenharmony_ci * Return a pointer to the ntfs_inode structure on success or NULL on error, 155987da915Sopenharmony_ci * with errno set to the error code. 156987da915Sopenharmony_ci */ 157987da915Sopenharmony_cistatic ntfs_inode *ntfs_inode_real_open(ntfs_volume *vol, const MFT_REF mref) 158987da915Sopenharmony_ci{ 159987da915Sopenharmony_ci s64 l; 160987da915Sopenharmony_ci ntfs_inode *ni = NULL; 161987da915Sopenharmony_ci ntfs_attr_search_ctx *ctx; 162987da915Sopenharmony_ci STANDARD_INFORMATION *std_info; 163987da915Sopenharmony_ci le32 lthle; 164987da915Sopenharmony_ci int olderrno; 165987da915Sopenharmony_ci 166987da915Sopenharmony_ci ntfs_log_enter("Entering for inode %lld\n", (long long)MREF(mref)); 167987da915Sopenharmony_ci if (!vol) { 168987da915Sopenharmony_ci errno = EINVAL; 169987da915Sopenharmony_ci goto out; 170987da915Sopenharmony_ci } 171987da915Sopenharmony_ci ni = __ntfs_inode_allocate(vol); 172987da915Sopenharmony_ci if (!ni) 173987da915Sopenharmony_ci goto out; 174987da915Sopenharmony_ci if (ntfs_file_record_read(vol, mref, &ni->mrec, NULL)) 175987da915Sopenharmony_ci goto err_out; 176987da915Sopenharmony_ci if (!(ni->mrec->flags & MFT_RECORD_IN_USE)) { 177987da915Sopenharmony_ci errno = ENOENT; 178987da915Sopenharmony_ci goto err_out; 179987da915Sopenharmony_ci } 180987da915Sopenharmony_ci ni->mft_no = MREF(mref); 181987da915Sopenharmony_ci ctx = ntfs_attr_get_search_ctx(ni, NULL); 182987da915Sopenharmony_ci if (!ctx) 183987da915Sopenharmony_ci goto err_out; 184987da915Sopenharmony_ci /* Receive some basic information about inode. */ 185987da915Sopenharmony_ci if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, AT_UNNAMED, 186987da915Sopenharmony_ci 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { 187987da915Sopenharmony_ci if (!ni->mrec->base_mft_record) 188987da915Sopenharmony_ci ntfs_log_perror("No STANDARD_INFORMATION in base record" 189987da915Sopenharmony_ci " %lld", (long long)MREF(mref)); 190987da915Sopenharmony_ci goto put_err_out; 191987da915Sopenharmony_ci } 192987da915Sopenharmony_ci lthle = ctx->attr->value_length; 193987da915Sopenharmony_ci if (le32_to_cpu(lthle) < offsetof(STANDARD_INFORMATION, owner_id)) { 194987da915Sopenharmony_ci ntfs_log_error("Corrupt STANDARD_INFORMATION in base" 195987da915Sopenharmony_ci " record %lld\n", 196987da915Sopenharmony_ci (long long)MREF(mref)); 197987da915Sopenharmony_ci goto put_err_out; 198987da915Sopenharmony_ci } 199987da915Sopenharmony_ci std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + 200987da915Sopenharmony_ci le16_to_cpu(ctx->attr->value_offset)); 201987da915Sopenharmony_ci ni->flags = std_info->file_attributes; 202987da915Sopenharmony_ci ni->creation_time = std_info->creation_time; 203987da915Sopenharmony_ci ni->last_data_change_time = std_info->last_data_change_time; 204987da915Sopenharmony_ci ni->last_mft_change_time = std_info->last_mft_change_time; 205987da915Sopenharmony_ci ni->last_access_time = std_info->last_access_time; 206987da915Sopenharmony_ci /* Insert v3 extensions if present */ 207987da915Sopenharmony_ci /* length may be seen as 48 (v1.x) or 72 (v3.x) */ 208987da915Sopenharmony_ci if (le32_to_cpu(lthle) >= offsetof(STANDARD_INFORMATION, v3_end)) { 209987da915Sopenharmony_ci set_nino_flag(ni, v3_Extensions); 210987da915Sopenharmony_ci ni->owner_id = std_info->owner_id; 211987da915Sopenharmony_ci ni->security_id = std_info->security_id; 212987da915Sopenharmony_ci ni->quota_charged = std_info->quota_charged; 213987da915Sopenharmony_ci ni->usn = std_info->usn; 214987da915Sopenharmony_ci } else { 215987da915Sopenharmony_ci clear_nino_flag(ni, v3_Extensions); 216987da915Sopenharmony_ci ni->owner_id = const_cpu_to_le32(0); 217987da915Sopenharmony_ci ni->security_id = const_cpu_to_le32(0); 218987da915Sopenharmony_ci } 219987da915Sopenharmony_ci /* Set attribute list information. */ 220987da915Sopenharmony_ci olderrno = errno; 221987da915Sopenharmony_ci if (ntfs_attr_lookup(AT_ATTRIBUTE_LIST, AT_UNNAMED, 0, 222987da915Sopenharmony_ci CASE_SENSITIVE, 0, NULL, 0, ctx)) { 223987da915Sopenharmony_ci if (errno != ENOENT) 224987da915Sopenharmony_ci goto put_err_out; 225987da915Sopenharmony_ci /* Attribute list attribute does not present. */ 226987da915Sopenharmony_ci /* restore previous errno to avoid misinterpretation */ 227987da915Sopenharmony_ci errno = olderrno; 228987da915Sopenharmony_ci goto get_size; 229987da915Sopenharmony_ci } 230987da915Sopenharmony_ci NInoSetAttrList(ni); 231987da915Sopenharmony_ci l = ntfs_get_attribute_value_length(ctx->attr); 232987da915Sopenharmony_ci if (!l) 233987da915Sopenharmony_ci goto put_err_out; 234987da915Sopenharmony_ci if ((u64)l > 0x40000) { 235987da915Sopenharmony_ci errno = EIO; 236987da915Sopenharmony_ci ntfs_log_perror("Too large attrlist attribute (%llu), inode " 237987da915Sopenharmony_ci "%lld", (long long)l, (long long)MREF(mref)); 238987da915Sopenharmony_ci goto put_err_out; 239987da915Sopenharmony_ci } 240987da915Sopenharmony_ci ni->attr_list_size = l; 241987da915Sopenharmony_ci ni->attr_list = ntfs_malloc(ni->attr_list_size); 242987da915Sopenharmony_ci if (!ni->attr_list) 243987da915Sopenharmony_ci goto put_err_out; 244987da915Sopenharmony_ci l = ntfs_get_attribute_value(vol, ctx->attr, ni->attr_list); 245987da915Sopenharmony_ci if (!l) 246987da915Sopenharmony_ci goto put_err_out; 247987da915Sopenharmony_ci if (l != ni->attr_list_size) { 248987da915Sopenharmony_ci errno = EIO; 249987da915Sopenharmony_ci ntfs_log_perror("Unexpected attrlist size (%lld <> %u), inode " 250987da915Sopenharmony_ci "%lld", (long long)l, ni->attr_list_size, 251987da915Sopenharmony_ci (long long)MREF(mref)); 252987da915Sopenharmony_ci goto put_err_out; 253987da915Sopenharmony_ci } 254987da915Sopenharmony_ciget_size: 255987da915Sopenharmony_ci olderrno = errno; 256987da915Sopenharmony_ci if (ntfs_attr_lookup(AT_DATA, AT_UNNAMED, 0, 0, 0, NULL, 0, ctx)) { 257987da915Sopenharmony_ci if (errno != ENOENT) 258987da915Sopenharmony_ci goto put_err_out; 259987da915Sopenharmony_ci /* Directory or special file. */ 260987da915Sopenharmony_ci /* restore previous errno to avoid misinterpretation */ 261987da915Sopenharmony_ci errno = olderrno; 262987da915Sopenharmony_ci ni->data_size = ni->allocated_size = 0; 263987da915Sopenharmony_ci } else { 264987da915Sopenharmony_ci if (ctx->attr->non_resident) { 265987da915Sopenharmony_ci ni->data_size = sle64_to_cpu(ctx->attr->data_size); 266987da915Sopenharmony_ci if (ctx->attr->flags & 267987da915Sopenharmony_ci (ATTR_IS_COMPRESSED | ATTR_IS_SPARSE)) 268987da915Sopenharmony_ci ni->allocated_size = sle64_to_cpu( 269987da915Sopenharmony_ci ctx->attr->compressed_size); 270987da915Sopenharmony_ci else 271987da915Sopenharmony_ci ni->allocated_size = sle64_to_cpu( 272987da915Sopenharmony_ci ctx->attr->allocated_size); 273987da915Sopenharmony_ci } else { 274987da915Sopenharmony_ci ni->data_size = le32_to_cpu(ctx->attr->value_length); 275987da915Sopenharmony_ci ni->allocated_size = (ni->data_size + 7) & ~7; 276987da915Sopenharmony_ci } 277987da915Sopenharmony_ci set_nino_flag(ni,KnownSize); 278987da915Sopenharmony_ci } 279987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 280987da915Sopenharmony_ciout: 281987da915Sopenharmony_ci ntfs_log_leave("\n"); 282987da915Sopenharmony_ci return ni; 283987da915Sopenharmony_ci 284987da915Sopenharmony_ciput_err_out: 285987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 286987da915Sopenharmony_cierr_out: 287987da915Sopenharmony_ci __ntfs_inode_release(ni); 288987da915Sopenharmony_ci ni = NULL; 289987da915Sopenharmony_ci goto out; 290987da915Sopenharmony_ci} 291987da915Sopenharmony_ci 292987da915Sopenharmony_ci/** 293987da915Sopenharmony_ci * ntfs_inode_close - close an ntfs inode and free all associated memory 294987da915Sopenharmony_ci * @ni: ntfs inode to close 295987da915Sopenharmony_ci * 296987da915Sopenharmony_ci * Make sure the ntfs inode @ni is clean. 297987da915Sopenharmony_ci * 298987da915Sopenharmony_ci * If the ntfs inode @ni is a base inode, close all associated extent inodes, 299987da915Sopenharmony_ci * then deallocate all memory attached to it, and finally free the ntfs inode 300987da915Sopenharmony_ci * structure itself. 301987da915Sopenharmony_ci * 302987da915Sopenharmony_ci * If it is an extent inode, we disconnect it from its base inode before we 303987da915Sopenharmony_ci * destroy it. 304987da915Sopenharmony_ci * 305987da915Sopenharmony_ci * It is OK to pass NULL to this function, it is just noop in this case. 306987da915Sopenharmony_ci * 307987da915Sopenharmony_ci * Return 0 on success or -1 on error with errno set to the error code. On 308987da915Sopenharmony_ci * error, @ni has not been freed. The user should attempt to handle the error 309987da915Sopenharmony_ci * and call ntfs_inode_close() again. The following error codes are defined: 310987da915Sopenharmony_ci * 311987da915Sopenharmony_ci * EBUSY @ni and/or its attribute list runlist is/are dirty and the 312987da915Sopenharmony_ci * attempt to write it/them to disk failed. 313987da915Sopenharmony_ci * EINVAL @ni is invalid (probably it is an extent inode). 314987da915Sopenharmony_ci * EIO I/O error while trying to write inode to disk. 315987da915Sopenharmony_ci */ 316987da915Sopenharmony_ci 317987da915Sopenharmony_ciint ntfs_inode_real_close(ntfs_inode *ni) 318987da915Sopenharmony_ci{ 319987da915Sopenharmony_ci int ret = -1; 320987da915Sopenharmony_ci 321987da915Sopenharmony_ci if (!ni) 322987da915Sopenharmony_ci return 0; 323987da915Sopenharmony_ci 324987da915Sopenharmony_ci ntfs_log_enter("Entering for inode %lld\n", (long long)ni->mft_no); 325987da915Sopenharmony_ci 326987da915Sopenharmony_ci /* If we have dirty metadata, write it out. */ 327987da915Sopenharmony_ci if (NInoDirty(ni) || NInoAttrListDirty(ni)) { 328987da915Sopenharmony_ci if (ntfs_inode_sync(ni)) { 329987da915Sopenharmony_ci if (errno != EIO) 330987da915Sopenharmony_ci errno = EBUSY; 331987da915Sopenharmony_ci goto err; 332987da915Sopenharmony_ci } 333987da915Sopenharmony_ci } 334987da915Sopenharmony_ci /* Is this a base inode with mapped extent inodes? */ 335987da915Sopenharmony_ci if (ni->nr_extents > 0) { 336987da915Sopenharmony_ci while (ni->nr_extents > 0) { 337987da915Sopenharmony_ci if (ntfs_inode_real_close(ni->extent_nis[0])) { 338987da915Sopenharmony_ci if (errno != EIO) 339987da915Sopenharmony_ci errno = EBUSY; 340987da915Sopenharmony_ci goto err; 341987da915Sopenharmony_ci } 342987da915Sopenharmony_ci } 343987da915Sopenharmony_ci } else if (ni->nr_extents == -1) { 344987da915Sopenharmony_ci ntfs_inode **tmp_nis; 345987da915Sopenharmony_ci ntfs_inode *base_ni; 346987da915Sopenharmony_ci s32 i; 347987da915Sopenharmony_ci 348987da915Sopenharmony_ci /* 349987da915Sopenharmony_ci * If the inode is an extent inode, disconnect it from the 350987da915Sopenharmony_ci * base inode before destroying it. 351987da915Sopenharmony_ci */ 352987da915Sopenharmony_ci base_ni = ni->base_ni; 353987da915Sopenharmony_ci for (i = 0; i < base_ni->nr_extents; ++i) { 354987da915Sopenharmony_ci tmp_nis = base_ni->extent_nis; 355987da915Sopenharmony_ci if (tmp_nis[i] != ni) 356987da915Sopenharmony_ci continue; 357987da915Sopenharmony_ci /* Found it. Disconnect. */ 358987da915Sopenharmony_ci memmove(tmp_nis + i, tmp_nis + i + 1, 359987da915Sopenharmony_ci (base_ni->nr_extents - i - 1) * 360987da915Sopenharmony_ci sizeof(ntfs_inode *)); 361987da915Sopenharmony_ci /* Buffer should be for multiple of four extents. */ 362987da915Sopenharmony_ci if ((--base_ni->nr_extents) & 3) { 363987da915Sopenharmony_ci i = -1; 364987da915Sopenharmony_ci break; 365987da915Sopenharmony_ci } 366987da915Sopenharmony_ci /* 367987da915Sopenharmony_ci * ElectricFence is unhappy with realloc(x,0) as free(x) 368987da915Sopenharmony_ci * thus we explicitly separate these two cases. 369987da915Sopenharmony_ci */ 370987da915Sopenharmony_ci if (base_ni->nr_extents) { 371987da915Sopenharmony_ci /* Resize the memory buffer. */ 372987da915Sopenharmony_ci tmp_nis = realloc(tmp_nis, base_ni->nr_extents * 373987da915Sopenharmony_ci sizeof(ntfs_inode *)); 374987da915Sopenharmony_ci /* Ignore errors, they don't really matter. */ 375987da915Sopenharmony_ci if (tmp_nis) 376987da915Sopenharmony_ci base_ni->extent_nis = tmp_nis; 377987da915Sopenharmony_ci } else if (tmp_nis) { 378987da915Sopenharmony_ci free(tmp_nis); 379987da915Sopenharmony_ci base_ni->extent_nis = (ntfs_inode**)NULL; 380987da915Sopenharmony_ci } 381987da915Sopenharmony_ci /* Allow for error checking. */ 382987da915Sopenharmony_ci i = -1; 383987da915Sopenharmony_ci break; 384987da915Sopenharmony_ci } 385987da915Sopenharmony_ci 386987da915Sopenharmony_ci /* 387987da915Sopenharmony_ci * We could successfully sync, so only log this error 388987da915Sopenharmony_ci * and try to sync other inode extents too. 389987da915Sopenharmony_ci */ 390987da915Sopenharmony_ci if (i != -1) 391987da915Sopenharmony_ci ntfs_log_error("Extent inode %lld was not found\n", 392987da915Sopenharmony_ci (long long)ni->mft_no); 393987da915Sopenharmony_ci } 394987da915Sopenharmony_ci 395987da915Sopenharmony_ci __ntfs_inode_release(ni); 396987da915Sopenharmony_ci ret = 0; 397987da915Sopenharmony_cierr: 398987da915Sopenharmony_ci ntfs_log_leave("\n"); 399987da915Sopenharmony_ci return ret; 400987da915Sopenharmony_ci} 401987da915Sopenharmony_ci 402987da915Sopenharmony_ci#if CACHE_NIDATA_SIZE 403987da915Sopenharmony_ci 404987da915Sopenharmony_ci/* 405987da915Sopenharmony_ci * Free an inode structure when there is not more space 406987da915Sopenharmony_ci * in the cache 407987da915Sopenharmony_ci */ 408987da915Sopenharmony_ci 409987da915Sopenharmony_civoid ntfs_inode_nidata_free(const struct CACHED_GENERIC *cached) 410987da915Sopenharmony_ci{ 411987da915Sopenharmony_ci ntfs_inode_real_close(((const struct CACHED_NIDATA*)cached)->ni); 412987da915Sopenharmony_ci} 413987da915Sopenharmony_ci 414987da915Sopenharmony_ci/* 415987da915Sopenharmony_ci * Compute a hash value for an inode entry 416987da915Sopenharmony_ci */ 417987da915Sopenharmony_ci 418987da915Sopenharmony_ciint ntfs_inode_nidata_hash(const struct CACHED_GENERIC *item) 419987da915Sopenharmony_ci{ 420987da915Sopenharmony_ci return (((const struct CACHED_NIDATA*)item)->inum 421987da915Sopenharmony_ci % (2*CACHE_NIDATA_SIZE)); 422987da915Sopenharmony_ci} 423987da915Sopenharmony_ci 424987da915Sopenharmony_ci/* 425987da915Sopenharmony_ci * inum comparing for entering/fetching from cache 426987da915Sopenharmony_ci */ 427987da915Sopenharmony_ci 428987da915Sopenharmony_cistatic int idata_cache_compare(const struct CACHED_GENERIC *cached, 429987da915Sopenharmony_ci const struct CACHED_GENERIC *wanted) 430987da915Sopenharmony_ci{ 431987da915Sopenharmony_ci return (((const struct CACHED_NIDATA*)cached)->inum 432987da915Sopenharmony_ci != ((const struct CACHED_NIDATA*)wanted)->inum); 433987da915Sopenharmony_ci} 434987da915Sopenharmony_ci 435987da915Sopenharmony_ci/* 436987da915Sopenharmony_ci * Invalidate an inode entry when not needed anymore. 437987da915Sopenharmony_ci * The entry should have been synced, it may be reused later, 438987da915Sopenharmony_ci * if it is requested before it is dropped from cache. 439987da915Sopenharmony_ci */ 440987da915Sopenharmony_ci 441987da915Sopenharmony_civoid ntfs_inode_invalidate(ntfs_volume *vol, const MFT_REF mref) 442987da915Sopenharmony_ci{ 443987da915Sopenharmony_ci struct CACHED_NIDATA item; 444987da915Sopenharmony_ci 445987da915Sopenharmony_ci item.inum = MREF(mref); 446987da915Sopenharmony_ci item.ni = (ntfs_inode*)NULL; 447987da915Sopenharmony_ci item.pathname = (const char*)NULL; 448987da915Sopenharmony_ci item.varsize = 0; 449987da915Sopenharmony_ci ntfs_invalidate_cache(vol->nidata_cache, 450987da915Sopenharmony_ci GENERIC(&item),idata_cache_compare,CACHE_FREE); 451987da915Sopenharmony_ci} 452987da915Sopenharmony_ci 453987da915Sopenharmony_ci#endif 454987da915Sopenharmony_ci 455987da915Sopenharmony_ci/* 456987da915Sopenharmony_ci * Open an inode 457987da915Sopenharmony_ci * 458987da915Sopenharmony_ci * When possible, an entry recorded in the cache is reused 459987da915Sopenharmony_ci * 460987da915Sopenharmony_ci * **NEVER REOPEN** an inode, this can lead to a duplicated 461987da915Sopenharmony_ci * cache entry (hard to detect), and to an obsolete one being 462987da915Sopenharmony_ci * reused. System files are however protected from being cached. 463987da915Sopenharmony_ci */ 464987da915Sopenharmony_ci 465987da915Sopenharmony_cintfs_inode *ntfs_inode_open(ntfs_volume *vol, const MFT_REF mref) 466987da915Sopenharmony_ci{ 467987da915Sopenharmony_ci ntfs_inode *ni; 468987da915Sopenharmony_ci#if CACHE_NIDATA_SIZE 469987da915Sopenharmony_ci struct CACHED_NIDATA item; 470987da915Sopenharmony_ci struct CACHED_NIDATA *cached; 471987da915Sopenharmony_ci 472987da915Sopenharmony_ci /* fetch idata from cache */ 473987da915Sopenharmony_ci item.inum = MREF(mref); 474987da915Sopenharmony_ci debug_double_inode(item.inum,1); 475987da915Sopenharmony_ci item.pathname = (const char*)NULL; 476987da915Sopenharmony_ci item.varsize = 0; 477987da915Sopenharmony_ci cached = (struct CACHED_NIDATA*)ntfs_fetch_cache(vol->nidata_cache, 478987da915Sopenharmony_ci GENERIC(&item),idata_cache_compare); 479987da915Sopenharmony_ci if (cached) { 480987da915Sopenharmony_ci ni = cached->ni; 481987da915Sopenharmony_ci /* do not keep open entries in cache */ 482987da915Sopenharmony_ci ntfs_remove_cache(vol->nidata_cache, 483987da915Sopenharmony_ci (struct CACHED_GENERIC*)cached,0); 484987da915Sopenharmony_ci } else { 485987da915Sopenharmony_ci ni = ntfs_inode_real_open(vol, mref); 486987da915Sopenharmony_ci } 487987da915Sopenharmony_ci if (!ni) { 488987da915Sopenharmony_ci debug_double_inode(item.inum, 0); 489987da915Sopenharmony_ci } 490987da915Sopenharmony_ci#else 491987da915Sopenharmony_ci ni = ntfs_inode_real_open(vol, mref); 492987da915Sopenharmony_ci#endif 493987da915Sopenharmony_ci return (ni); 494987da915Sopenharmony_ci} 495987da915Sopenharmony_ci 496987da915Sopenharmony_ci/* 497987da915Sopenharmony_ci * Close an inode entry 498987da915Sopenharmony_ci * 499987da915Sopenharmony_ci * If cacheing is in use, the entry is synced and kept available 500987da915Sopenharmony_ci * in cache for further use. 501987da915Sopenharmony_ci * 502987da915Sopenharmony_ci * System files (inode < 16 or having the IS_4 flag) are protected 503987da915Sopenharmony_ci * against being cached. 504987da915Sopenharmony_ci */ 505987da915Sopenharmony_ci 506987da915Sopenharmony_ciint ntfs_inode_close(ntfs_inode *ni) 507987da915Sopenharmony_ci{ 508987da915Sopenharmony_ci int res; 509987da915Sopenharmony_ci#if CACHE_NIDATA_SIZE 510987da915Sopenharmony_ci BOOL dirty; 511987da915Sopenharmony_ci struct CACHED_NIDATA item; 512987da915Sopenharmony_ci 513987da915Sopenharmony_ci if (ni) { 514987da915Sopenharmony_ci debug_double_inode(ni->mft_no,0); 515987da915Sopenharmony_ci /* do not cache system files : could lead to double entries */ 516987da915Sopenharmony_ci if (ni->vol && ni->vol->nidata_cache 517987da915Sopenharmony_ci && ((ni->mft_no == FILE_root) 518987da915Sopenharmony_ci || ((ni->mft_no >= FILE_first_user) 519987da915Sopenharmony_ci && !(ni->mrec->flags & MFT_RECORD_IS_4)))) { 520987da915Sopenharmony_ci /* If we have dirty metadata, write it out. */ 521987da915Sopenharmony_ci dirty = NInoDirty(ni) || NInoAttrListDirty(ni); 522987da915Sopenharmony_ci if (dirty) { 523987da915Sopenharmony_ci res = ntfs_inode_sync(ni); 524987da915Sopenharmony_ci /* do a real close if sync failed */ 525987da915Sopenharmony_ci if (res) 526987da915Sopenharmony_ci ntfs_inode_real_close(ni); 527987da915Sopenharmony_ci } else 528987da915Sopenharmony_ci res = 0; 529987da915Sopenharmony_ci 530987da915Sopenharmony_ci if (!res) { 531987da915Sopenharmony_ci /* feed idata into cache */ 532987da915Sopenharmony_ci item.inum = ni->mft_no; 533987da915Sopenharmony_ci item.ni = ni; 534987da915Sopenharmony_ci item.pathname = (const char*)NULL; 535987da915Sopenharmony_ci item.varsize = 0; 536987da915Sopenharmony_ci debug_cached_inode(ni); 537987da915Sopenharmony_ci ntfs_enter_cache(ni->vol->nidata_cache, 538987da915Sopenharmony_ci GENERIC(&item), idata_cache_compare); 539987da915Sopenharmony_ci } 540987da915Sopenharmony_ci } else { 541987da915Sopenharmony_ci /* cache not ready or system file, really close */ 542987da915Sopenharmony_ci res = ntfs_inode_real_close(ni); 543987da915Sopenharmony_ci } 544987da915Sopenharmony_ci } else 545987da915Sopenharmony_ci res = 0; 546987da915Sopenharmony_ci#else 547987da915Sopenharmony_ci res = ntfs_inode_real_close(ni); 548987da915Sopenharmony_ci#endif 549987da915Sopenharmony_ci return (res); 550987da915Sopenharmony_ci} 551987da915Sopenharmony_ci 552987da915Sopenharmony_ci/** 553987da915Sopenharmony_ci * ntfs_extent_inode_open - load an extent inode and attach it to its base 554987da915Sopenharmony_ci * @base_ni: base ntfs inode 555987da915Sopenharmony_ci * @mref: mft reference of the extent inode to load (in little endian) 556987da915Sopenharmony_ci * 557987da915Sopenharmony_ci * First check if the extent inode @mref is already attached to the base ntfs 558987da915Sopenharmony_ci * inode @base_ni, and if so, return a pointer to the attached extent inode. 559987da915Sopenharmony_ci * 560987da915Sopenharmony_ci * If the extent inode is not already attached to the base inode, allocate an 561987da915Sopenharmony_ci * ntfs_inode structure and initialize it for the given inode @mref. @mref 562987da915Sopenharmony_ci * specifies the inode number / mft record to read, including the sequence 563987da915Sopenharmony_ci * number, which can be 0 if no sequence number checking is to be performed. 564987da915Sopenharmony_ci * 565987da915Sopenharmony_ci * Then, allocate a buffer for the mft record, read the mft record from the 566987da915Sopenharmony_ci * volume @base_ni->vol, and attach it to the ntfs_inode structure (->mrec). 567987da915Sopenharmony_ci * The mft record is mst deprotected and sanity checked for validity and we 568987da915Sopenharmony_ci * abort if deprotection or checks fail. 569987da915Sopenharmony_ci * 570987da915Sopenharmony_ci * Finally attach the ntfs inode to its base inode @base_ni and return a 571987da915Sopenharmony_ci * pointer to the ntfs_inode structure on success or NULL on error, with errno 572987da915Sopenharmony_ci * set to the error code. 573987da915Sopenharmony_ci * 574987da915Sopenharmony_ci * Note, extent inodes are never closed directly. They are automatically 575987da915Sopenharmony_ci * disposed off by the closing of the base inode. 576987da915Sopenharmony_ci */ 577987da915Sopenharmony_cintfs_inode *ntfs_extent_inode_open(ntfs_inode *base_ni, const leMFT_REF mref) 578987da915Sopenharmony_ci{ 579987da915Sopenharmony_ci u64 mft_no = MREF_LE(mref); 580987da915Sopenharmony_ci VCN extent_vcn; 581987da915Sopenharmony_ci runlist_element *rl; 582987da915Sopenharmony_ci ntfs_volume *vol; 583987da915Sopenharmony_ci ntfs_inode *ni = NULL; 584987da915Sopenharmony_ci ntfs_inode **extent_nis; 585987da915Sopenharmony_ci int i; 586987da915Sopenharmony_ci 587987da915Sopenharmony_ci if (!base_ni) { 588987da915Sopenharmony_ci errno = EINVAL; 589987da915Sopenharmony_ci ntfs_log_perror("%s", __FUNCTION__); 590987da915Sopenharmony_ci return NULL; 591987da915Sopenharmony_ci } 592987da915Sopenharmony_ci 593987da915Sopenharmony_ci ntfs_log_enter("Opening extent inode %lld (base mft record %lld).\n", 594987da915Sopenharmony_ci (unsigned long long)mft_no, 595987da915Sopenharmony_ci (unsigned long long)base_ni->mft_no); 596987da915Sopenharmony_ci 597987da915Sopenharmony_ci if (!base_ni->mft_no) { 598987da915Sopenharmony_ci /* 599987da915Sopenharmony_ci * When getting extents of MFT, we must be sure 600987da915Sopenharmony_ci * they are in the MFT part which has already 601987da915Sopenharmony_ci * been mapped, otherwise we fall into an endless 602987da915Sopenharmony_ci * recursion. 603987da915Sopenharmony_ci * Situations have been met where extents locations 604987da915Sopenharmony_ci * are described in themselves. 605987da915Sopenharmony_ci * This is a severe error which chkdsk cannot fix. 606987da915Sopenharmony_ci */ 607987da915Sopenharmony_ci vol = base_ni->vol; 608987da915Sopenharmony_ci extent_vcn = mft_no << vol->mft_record_size_bits 609987da915Sopenharmony_ci >> vol->cluster_size_bits; 610987da915Sopenharmony_ci rl = vol->mft_na->rl; 611987da915Sopenharmony_ci if (rl) { 612987da915Sopenharmony_ci while (rl->length 613987da915Sopenharmony_ci && ((rl->vcn + rl->length) <= extent_vcn)) 614987da915Sopenharmony_ci rl++; 615987da915Sopenharmony_ci } 616987da915Sopenharmony_ci if (!rl || (rl->lcn < 0)) { 617987da915Sopenharmony_ci ntfs_log_error("MFT is corrupt, cannot read" 618987da915Sopenharmony_ci " its unmapped extent record %lld\n", 619987da915Sopenharmony_ci (long long)mft_no); 620987da915Sopenharmony_ci ntfs_log_error("Note : chkdsk cannot fix this," 621987da915Sopenharmony_ci " try ntfsfix\n"); 622987da915Sopenharmony_ci errno = EIO; 623987da915Sopenharmony_ci ni = (ntfs_inode*)NULL; 624987da915Sopenharmony_ci goto out; 625987da915Sopenharmony_ci } 626987da915Sopenharmony_ci } 627987da915Sopenharmony_ci 628987da915Sopenharmony_ci /* Is the extent inode already open and attached to the base inode? */ 629987da915Sopenharmony_ci if (base_ni->nr_extents > 0) { 630987da915Sopenharmony_ci extent_nis = base_ni->extent_nis; 631987da915Sopenharmony_ci for (i = 0; i < base_ni->nr_extents; i++) { 632987da915Sopenharmony_ci u16 seq_no; 633987da915Sopenharmony_ci 634987da915Sopenharmony_ci ni = extent_nis[i]; 635987da915Sopenharmony_ci if (mft_no != ni->mft_no) 636987da915Sopenharmony_ci continue; 637987da915Sopenharmony_ci /* Verify the sequence number if given. */ 638987da915Sopenharmony_ci seq_no = MSEQNO_LE(mref); 639987da915Sopenharmony_ci if (seq_no && seq_no != le16_to_cpu( 640987da915Sopenharmony_ci ni->mrec->sequence_number)) { 641987da915Sopenharmony_ci errno = EIO; 642987da915Sopenharmony_ci ntfs_log_perror("Found stale extent mft " 643987da915Sopenharmony_ci "reference mft=%lld", 644987da915Sopenharmony_ci (long long)ni->mft_no); 645987da915Sopenharmony_ci goto out; 646987da915Sopenharmony_ci } 647987da915Sopenharmony_ci goto out; 648987da915Sopenharmony_ci } 649987da915Sopenharmony_ci } 650987da915Sopenharmony_ci /* Wasn't there, we need to load the extent inode. */ 651987da915Sopenharmony_ci ni = __ntfs_inode_allocate(base_ni->vol); 652987da915Sopenharmony_ci if (!ni) 653987da915Sopenharmony_ci goto out; 654987da915Sopenharmony_ci if (ntfs_file_record_read(base_ni->vol, le64_to_cpu(mref), &ni->mrec, NULL)) 655987da915Sopenharmony_ci goto err_out; 656987da915Sopenharmony_ci ni->mft_no = mft_no; 657987da915Sopenharmony_ci ni->nr_extents = -1; 658987da915Sopenharmony_ci ni->base_ni = base_ni; 659987da915Sopenharmony_ci /* Attach extent inode to base inode, reallocating memory if needed. */ 660987da915Sopenharmony_ci if (!(base_ni->nr_extents & 3)) { 661987da915Sopenharmony_ci i = (base_ni->nr_extents + 4) * sizeof(ntfs_inode *); 662987da915Sopenharmony_ci 663987da915Sopenharmony_ci extent_nis = ntfs_malloc(i); 664987da915Sopenharmony_ci if (!extent_nis) 665987da915Sopenharmony_ci goto err_out; 666987da915Sopenharmony_ci if (base_ni->nr_extents) { 667987da915Sopenharmony_ci memcpy(extent_nis, base_ni->extent_nis, 668987da915Sopenharmony_ci i - 4 * sizeof(ntfs_inode *)); 669987da915Sopenharmony_ci free(base_ni->extent_nis); 670987da915Sopenharmony_ci } 671987da915Sopenharmony_ci base_ni->extent_nis = extent_nis; 672987da915Sopenharmony_ci } 673987da915Sopenharmony_ci base_ni->extent_nis[base_ni->nr_extents++] = ni; 674987da915Sopenharmony_ciout: 675987da915Sopenharmony_ci ntfs_log_leave("\n"); 676987da915Sopenharmony_ci return ni; 677987da915Sopenharmony_cierr_out: 678987da915Sopenharmony_ci __ntfs_inode_release(ni); 679987da915Sopenharmony_ci ni = NULL; 680987da915Sopenharmony_ci goto out; 681987da915Sopenharmony_ci} 682987da915Sopenharmony_ci 683987da915Sopenharmony_ci/** 684987da915Sopenharmony_ci * ntfs_inode_attach_all_extents - attach all extents for target inode 685987da915Sopenharmony_ci * @ni: opened ntfs inode for which perform attach 686987da915Sopenharmony_ci * 687987da915Sopenharmony_ci * Return 0 on success and -1 on error with errno set to the error code. 688987da915Sopenharmony_ci */ 689987da915Sopenharmony_ciint ntfs_inode_attach_all_extents(ntfs_inode *ni) 690987da915Sopenharmony_ci{ 691987da915Sopenharmony_ci ATTR_LIST_ENTRY *ale; 692987da915Sopenharmony_ci u64 prev_attached = 0; 693987da915Sopenharmony_ci 694987da915Sopenharmony_ci if (!ni) { 695987da915Sopenharmony_ci ntfs_log_trace("Invalid arguments.\n"); 696987da915Sopenharmony_ci errno = EINVAL; 697987da915Sopenharmony_ci return -1; 698987da915Sopenharmony_ci } 699987da915Sopenharmony_ci 700987da915Sopenharmony_ci if (ni->nr_extents == -1) 701987da915Sopenharmony_ci ni = ni->base_ni; 702987da915Sopenharmony_ci 703987da915Sopenharmony_ci ntfs_log_trace("Entering for inode 0x%llx.\n", (long long) ni->mft_no); 704987da915Sopenharmony_ci 705987da915Sopenharmony_ci /* Inode haven't got attribute list, thus nothing to attach. */ 706987da915Sopenharmony_ci if (!NInoAttrList(ni)) 707987da915Sopenharmony_ci return 0; 708987da915Sopenharmony_ci 709987da915Sopenharmony_ci if (!ni->attr_list) { 710987da915Sopenharmony_ci ntfs_log_trace("Corrupt in-memory struct.\n"); 711987da915Sopenharmony_ci errno = EINVAL; 712987da915Sopenharmony_ci return -1; 713987da915Sopenharmony_ci } 714987da915Sopenharmony_ci 715987da915Sopenharmony_ci /* Walk through attribute list and attach all extents. */ 716987da915Sopenharmony_ci errno = 0; 717987da915Sopenharmony_ci ale = (ATTR_LIST_ENTRY *)ni->attr_list; 718987da915Sopenharmony_ci while ((u8*)ale < ni->attr_list + ni->attr_list_size) { 719987da915Sopenharmony_ci if (ni->mft_no != MREF_LE(ale->mft_reference) && 720987da915Sopenharmony_ci prev_attached != MREF_LE(ale->mft_reference)) { 721987da915Sopenharmony_ci if (!ntfs_extent_inode_open(ni, ale->mft_reference)) { 722987da915Sopenharmony_ci ntfs_log_trace("Couldn't attach extent inode.\n"); 723987da915Sopenharmony_ci return -1; 724987da915Sopenharmony_ci } 725987da915Sopenharmony_ci prev_attached = MREF_LE(ale->mft_reference); 726987da915Sopenharmony_ci } 727987da915Sopenharmony_ci ale = (ATTR_LIST_ENTRY *)((u8*)ale + le16_to_cpu(ale->length)); 728987da915Sopenharmony_ci } 729987da915Sopenharmony_ci return 0; 730987da915Sopenharmony_ci} 731987da915Sopenharmony_ci 732987da915Sopenharmony_ci/** 733987da915Sopenharmony_ci * ntfs_inode_sync_standard_information - update standard information attribute 734987da915Sopenharmony_ci * @ni: ntfs inode to update standard information 735987da915Sopenharmony_ci * 736987da915Sopenharmony_ci * Return 0 on success or -1 on error with errno set to the error code. 737987da915Sopenharmony_ci */ 738987da915Sopenharmony_cistatic int ntfs_inode_sync_standard_information(ntfs_inode *ni) 739987da915Sopenharmony_ci{ 740987da915Sopenharmony_ci ntfs_attr_search_ctx *ctx; 741987da915Sopenharmony_ci STANDARD_INFORMATION *std_info; 742987da915Sopenharmony_ci u32 lth; 743987da915Sopenharmony_ci le32 lthle; 744987da915Sopenharmony_ci 745987da915Sopenharmony_ci ntfs_log_trace("Entering for inode %lld\n", (long long)ni->mft_no); 746987da915Sopenharmony_ci 747987da915Sopenharmony_ci ctx = ntfs_attr_get_search_ctx(ni, NULL); 748987da915Sopenharmony_ci if (!ctx) 749987da915Sopenharmony_ci return -1; 750987da915Sopenharmony_ci if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, AT_UNNAMED, 751987da915Sopenharmony_ci 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { 752987da915Sopenharmony_ci ntfs_log_perror("Failed to sync standard info (inode %lld)", 753987da915Sopenharmony_ci (long long)ni->mft_no); 754987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 755987da915Sopenharmony_ci return -1; 756987da915Sopenharmony_ci } 757987da915Sopenharmony_ci std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + 758987da915Sopenharmony_ci le16_to_cpu(ctx->attr->value_offset)); 759987da915Sopenharmony_ci std_info->file_attributes = ni->flags; 760987da915Sopenharmony_ci if (!test_nino_flag(ni, TimesSet)) { 761987da915Sopenharmony_ci std_info->creation_time = ni->creation_time; 762987da915Sopenharmony_ci std_info->last_data_change_time = ni->last_data_change_time; 763987da915Sopenharmony_ci std_info->last_mft_change_time = ni->last_mft_change_time; 764987da915Sopenharmony_ci std_info->last_access_time = ni->last_access_time; 765987da915Sopenharmony_ci } 766987da915Sopenharmony_ci 767987da915Sopenharmony_ci /* JPA update v3.x extensions, ensuring consistency */ 768987da915Sopenharmony_ci 769987da915Sopenharmony_ci lthle = ctx->attr->value_length; 770987da915Sopenharmony_ci lth = le32_to_cpu(lthle); 771987da915Sopenharmony_ci if (test_nino_flag(ni, v3_Extensions) 772987da915Sopenharmony_ci && (lth < offsetof(STANDARD_INFORMATION, v3_end))) 773987da915Sopenharmony_ci ntfs_log_error("bad sync of standard information\n"); 774987da915Sopenharmony_ci 775987da915Sopenharmony_ci if (lth >= offsetof(STANDARD_INFORMATION, v3_end)) { 776987da915Sopenharmony_ci std_info->owner_id = ni->owner_id; 777987da915Sopenharmony_ci std_info->security_id = ni->security_id; 778987da915Sopenharmony_ci std_info->quota_charged = ni->quota_charged; 779987da915Sopenharmony_ci std_info->usn = ni->usn; 780987da915Sopenharmony_ci } 781987da915Sopenharmony_ci ntfs_inode_mark_dirty(ctx->ntfs_ino); 782987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 783987da915Sopenharmony_ci return 0; 784987da915Sopenharmony_ci} 785987da915Sopenharmony_ci 786987da915Sopenharmony_ci/** 787987da915Sopenharmony_ci * ntfs_inode_sync_file_name - update FILE_NAME attributes 788987da915Sopenharmony_ci * @ni: ntfs inode to update FILE_NAME attributes 789987da915Sopenharmony_ci * 790987da915Sopenharmony_ci * Update all FILE_NAME attributes for inode @ni in the index. 791987da915Sopenharmony_ci * 792987da915Sopenharmony_ci * Return 0 on success or -1 on error with errno set to the error code. 793987da915Sopenharmony_ci */ 794987da915Sopenharmony_cistatic int ntfs_inode_sync_file_name(ntfs_inode *ni, ntfs_inode *dir_ni) 795987da915Sopenharmony_ci{ 796987da915Sopenharmony_ci ntfs_attr_search_ctx *ctx = NULL; 797987da915Sopenharmony_ci ntfs_index_context *ictx; 798987da915Sopenharmony_ci ntfs_inode *index_ni; 799987da915Sopenharmony_ci FILE_NAME_ATTR *fn; 800987da915Sopenharmony_ci FILE_NAME_ATTR *fnx; 801987da915Sopenharmony_ci REPARSE_POINT *rpp; 802987da915Sopenharmony_ci le32 reparse_tag; 803987da915Sopenharmony_ci int err = 0; 804987da915Sopenharmony_ci 805987da915Sopenharmony_ci ntfs_log_trace("Entering for inode %lld\n", (long long)ni->mft_no); 806987da915Sopenharmony_ci 807987da915Sopenharmony_ci ctx = ntfs_attr_get_search_ctx(ni, NULL); 808987da915Sopenharmony_ci if (!ctx) { 809987da915Sopenharmony_ci err = errno; 810987da915Sopenharmony_ci goto err_out; 811987da915Sopenharmony_ci } 812987da915Sopenharmony_ci /* Collect the reparse tag, if any */ 813987da915Sopenharmony_ci reparse_tag = const_cpu_to_le32(0); 814987da915Sopenharmony_ci if (ni->flags & FILE_ATTR_REPARSE_POINT) { 815987da915Sopenharmony_ci if (!ntfs_attr_lookup(AT_REPARSE_POINT, NULL, 816987da915Sopenharmony_ci 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { 817987da915Sopenharmony_ci rpp = (REPARSE_POINT*)((u8 *)ctx->attr + 818987da915Sopenharmony_ci le16_to_cpu(ctx->attr->value_offset)); 819987da915Sopenharmony_ci reparse_tag = rpp->reparse_tag; 820987da915Sopenharmony_ci } 821987da915Sopenharmony_ci ntfs_attr_reinit_search_ctx(ctx); 822987da915Sopenharmony_ci } 823987da915Sopenharmony_ci /* Walk through all FILE_NAME attributes and update them. */ 824987da915Sopenharmony_ci while (!ntfs_attr_lookup(AT_FILE_NAME, NULL, 0, 0, 0, NULL, 0, ctx)) { 825987da915Sopenharmony_ci fn = (FILE_NAME_ATTR *)((u8 *)ctx->attr + 826987da915Sopenharmony_ci le16_to_cpu(ctx->attr->value_offset)); 827987da915Sopenharmony_ci if (MREF_LE(fn->parent_directory) == ni->mft_no) { 828987da915Sopenharmony_ci /* 829987da915Sopenharmony_ci * WARNING: We cheat here and obtain 2 attribute 830987da915Sopenharmony_ci * search contexts for one inode (first we obtained 831987da915Sopenharmony_ci * above, second will be obtained inside 832987da915Sopenharmony_ci * ntfs_index_lookup), it's acceptable for library, 833987da915Sopenharmony_ci * but will deadlock in the kernel. 834987da915Sopenharmony_ci */ 835987da915Sopenharmony_ci index_ni = ni; 836987da915Sopenharmony_ci } else 837987da915Sopenharmony_ci if (dir_ni) 838987da915Sopenharmony_ci index_ni = dir_ni; 839987da915Sopenharmony_ci else 840987da915Sopenharmony_ci index_ni = ntfs_inode_open(ni->vol, 841987da915Sopenharmony_ci le64_to_cpu(fn->parent_directory)); 842987da915Sopenharmony_ci if (!index_ni) { 843987da915Sopenharmony_ci if (!err) 844987da915Sopenharmony_ci err = errno; 845987da915Sopenharmony_ci ntfs_log_perror("Failed to open inode %lld with index", 846987da915Sopenharmony_ci (long long)MREF_LE(fn->parent_directory)); 847987da915Sopenharmony_ci continue; 848987da915Sopenharmony_ci } 849987da915Sopenharmony_ci ictx = ntfs_index_ctx_get(index_ni, NTFS_INDEX_I30, 4); 850987da915Sopenharmony_ci if (!ictx) { 851987da915Sopenharmony_ci if (!err) 852987da915Sopenharmony_ci err = errno; 853987da915Sopenharmony_ci ntfs_log_perror("Failed to get index ctx, inode %lld", 854987da915Sopenharmony_ci (long long)index_ni->mft_no); 855987da915Sopenharmony_ci if ((ni != index_ni) && !dir_ni 856987da915Sopenharmony_ci && ntfs_inode_close(index_ni) && !err) 857987da915Sopenharmony_ci err = errno; 858987da915Sopenharmony_ci continue; 859987da915Sopenharmony_ci } 860987da915Sopenharmony_ci if (ntfs_index_lookup(fn, sizeof(FILE_NAME_ATTR), ictx)) { 861987da915Sopenharmony_ci if (!err) { 862987da915Sopenharmony_ci if (errno == ENOENT) 863987da915Sopenharmony_ci err = EIO; 864987da915Sopenharmony_ci else 865987da915Sopenharmony_ci err = errno; 866987da915Sopenharmony_ci } 867987da915Sopenharmony_ci ntfs_log_perror("Index lookup failed, inode %lld", 868987da915Sopenharmony_ci (long long)index_ni->mft_no); 869987da915Sopenharmony_ci ntfs_index_ctx_put(ictx); 870987da915Sopenharmony_ci if (ni != index_ni && ntfs_inode_close(index_ni) && !err) 871987da915Sopenharmony_ci err = errno; 872987da915Sopenharmony_ci continue; 873987da915Sopenharmony_ci } 874987da915Sopenharmony_ci /* Update flags and file size. */ 875987da915Sopenharmony_ci fnx = (FILE_NAME_ATTR *)ictx->data; 876987da915Sopenharmony_ci fnx->file_attributes = 877987da915Sopenharmony_ci (fnx->file_attributes & ~FILE_ATTR_VALID_FLAGS) | 878987da915Sopenharmony_ci (ni->flags & FILE_ATTR_VALID_FLAGS); 879987da915Sopenharmony_ci if (ni->mrec->flags & MFT_RECORD_IS_DIRECTORY) 880987da915Sopenharmony_ci fnx->data_size = fnx->allocated_size 881987da915Sopenharmony_ci = const_cpu_to_sle64(0); 882987da915Sopenharmony_ci else { 883987da915Sopenharmony_ci fnx->allocated_size = cpu_to_sle64(ni->allocated_size); 884987da915Sopenharmony_ci fnx->data_size = cpu_to_sle64(ni->data_size); 885987da915Sopenharmony_ci /* 886987da915Sopenharmony_ci * The file name record has also to be fixed if some 887987da915Sopenharmony_ci * attribute update implied the unnamed data to be 888987da915Sopenharmony_ci * made non-resident 889987da915Sopenharmony_ci */ 890987da915Sopenharmony_ci fn->allocated_size = fnx->allocated_size; 891987da915Sopenharmony_ci } 892987da915Sopenharmony_ci /* update or clear the reparse tag in the index */ 893987da915Sopenharmony_ci fnx->reparse_point_tag = reparse_tag; 894987da915Sopenharmony_ci if (!test_nino_flag(ni, TimesSet)) { 895987da915Sopenharmony_ci fnx->creation_time = ni->creation_time; 896987da915Sopenharmony_ci fnx->last_data_change_time = ni->last_data_change_time; 897987da915Sopenharmony_ci fnx->last_mft_change_time = ni->last_mft_change_time; 898987da915Sopenharmony_ci fnx->last_access_time = ni->last_access_time; 899987da915Sopenharmony_ci } else { 900987da915Sopenharmony_ci fnx->creation_time = fn->creation_time; 901987da915Sopenharmony_ci fnx->last_data_change_time = fn->last_data_change_time; 902987da915Sopenharmony_ci fnx->last_mft_change_time = fn->last_mft_change_time; 903987da915Sopenharmony_ci fnx->last_access_time = fn->last_access_time; 904987da915Sopenharmony_ci } 905987da915Sopenharmony_ci ntfs_index_entry_mark_dirty(ictx); 906987da915Sopenharmony_ci ntfs_index_ctx_put(ictx); 907987da915Sopenharmony_ci if ((ni != index_ni) && !dir_ni 908987da915Sopenharmony_ci && ntfs_inode_close(index_ni) && !err) 909987da915Sopenharmony_ci err = errno; 910987da915Sopenharmony_ci } 911987da915Sopenharmony_ci /* Check for real error occurred. */ 912987da915Sopenharmony_ci if (errno != ENOENT) { 913987da915Sopenharmony_ci err = errno; 914987da915Sopenharmony_ci ntfs_log_perror("Attribute lookup failed, inode %lld", 915987da915Sopenharmony_ci (long long)ni->mft_no); 916987da915Sopenharmony_ci goto err_out; 917987da915Sopenharmony_ci } 918987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 919987da915Sopenharmony_ci if (err) { 920987da915Sopenharmony_ci errno = err; 921987da915Sopenharmony_ci return -1; 922987da915Sopenharmony_ci } 923987da915Sopenharmony_ci return 0; 924987da915Sopenharmony_cierr_out: 925987da915Sopenharmony_ci if (ctx) 926987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 927987da915Sopenharmony_ci errno = err; 928987da915Sopenharmony_ci return -1; 929987da915Sopenharmony_ci} 930987da915Sopenharmony_ci 931987da915Sopenharmony_ci/** 932987da915Sopenharmony_ci * ntfs_inode_sync - write the inode (and its dirty extents) to disk 933987da915Sopenharmony_ci * @ni: ntfs inode to write 934987da915Sopenharmony_ci * 935987da915Sopenharmony_ci * Write the inode @ni to disk as well as its dirty extent inodes if such 936987da915Sopenharmony_ci * exist and @ni is a base inode. If @ni is an extent inode, only @ni is 937987da915Sopenharmony_ci * written completely disregarding its base inode and any other extent inodes. 938987da915Sopenharmony_ci * 939987da915Sopenharmony_ci * For a base inode with dirty extent inodes if any writes fail for whatever 940987da915Sopenharmony_ci * reason, the failing inode is skipped and the sync process is continued. At 941987da915Sopenharmony_ci * the end the error condition that brought about the failure is returned. Thus 942987da915Sopenharmony_ci * the smallest amount of data loss possible occurs. 943987da915Sopenharmony_ci * 944987da915Sopenharmony_ci * Return 0 on success or -1 on error with errno set to the error code. 945987da915Sopenharmony_ci * The following error codes are defined: 946987da915Sopenharmony_ci * EINVAL - Invalid arguments were passed to the function. 947987da915Sopenharmony_ci * EBUSY - Inode and/or one of its extents is busy, try again later. 948987da915Sopenharmony_ci * EIO - I/O error while writing the inode (or one of its extents). 949987da915Sopenharmony_ci */ 950987da915Sopenharmony_cistatic int ntfs_inode_sync_in_dir(ntfs_inode *ni, ntfs_inode *dir_ni) 951987da915Sopenharmony_ci{ 952987da915Sopenharmony_ci int ret = 0; 953987da915Sopenharmony_ci int err = 0; 954987da915Sopenharmony_ci if (!ni) { 955987da915Sopenharmony_ci errno = EINVAL; 956987da915Sopenharmony_ci ntfs_log_error("Failed to sync NULL inode\n"); 957987da915Sopenharmony_ci return -1; 958987da915Sopenharmony_ci } 959987da915Sopenharmony_ci 960987da915Sopenharmony_ci ntfs_log_enter("Entering for inode %lld\n", (long long)ni->mft_no); 961987da915Sopenharmony_ci 962987da915Sopenharmony_ci /* Update STANDARD_INFORMATION. */ 963987da915Sopenharmony_ci if ((ni->mrec->flags & MFT_RECORD_IN_USE) && ni->nr_extents != -1 && 964987da915Sopenharmony_ci ntfs_inode_sync_standard_information(ni)) { 965987da915Sopenharmony_ci if (!err || errno == EIO) { 966987da915Sopenharmony_ci err = errno; 967987da915Sopenharmony_ci if (err != EIO) 968987da915Sopenharmony_ci err = EBUSY; 969987da915Sopenharmony_ci } 970987da915Sopenharmony_ci } 971987da915Sopenharmony_ci 972987da915Sopenharmony_ci /* Update FILE_NAME's in the index. */ 973987da915Sopenharmony_ci if ((ni->mrec->flags & MFT_RECORD_IN_USE) && ni->nr_extents != -1 && 974987da915Sopenharmony_ci NInoFileNameTestAndClearDirty(ni) && 975987da915Sopenharmony_ci ntfs_inode_sync_file_name(ni, dir_ni)) { 976987da915Sopenharmony_ci if (!err || errno == EIO) { 977987da915Sopenharmony_ci err = errno; 978987da915Sopenharmony_ci if (err != EIO) 979987da915Sopenharmony_ci err = EBUSY; 980987da915Sopenharmony_ci } 981987da915Sopenharmony_ci ntfs_log_perror("Failed to sync FILE_NAME (inode %lld)", 982987da915Sopenharmony_ci (long long)ni->mft_no); 983987da915Sopenharmony_ci NInoFileNameSetDirty(ni); 984987da915Sopenharmony_ci } 985987da915Sopenharmony_ci 986987da915Sopenharmony_ci /* Write out attribute list from cache to disk. */ 987987da915Sopenharmony_ci if ((ni->mrec->flags & MFT_RECORD_IN_USE) && ni->nr_extents != -1 && 988987da915Sopenharmony_ci NInoAttrList(ni) && NInoAttrListTestAndClearDirty(ni)) { 989987da915Sopenharmony_ci ntfs_attr *na; 990987da915Sopenharmony_ci 991987da915Sopenharmony_ci na = ntfs_attr_open(ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0); 992987da915Sopenharmony_ci if (!na) { 993987da915Sopenharmony_ci if (!err || errno == EIO) { 994987da915Sopenharmony_ci err = errno; 995987da915Sopenharmony_ci if (err != EIO) 996987da915Sopenharmony_ci err = EBUSY; 997987da915Sopenharmony_ci ntfs_log_perror("Attribute list sync failed " 998987da915Sopenharmony_ci "(open, inode %lld)", 999987da915Sopenharmony_ci (long long)ni->mft_no); 1000987da915Sopenharmony_ci } 1001987da915Sopenharmony_ci NInoAttrListSetDirty(ni); 1002987da915Sopenharmony_ci goto sync_inode; 1003987da915Sopenharmony_ci } 1004987da915Sopenharmony_ci 1005987da915Sopenharmony_ci if (na->data_size == ni->attr_list_size) { 1006987da915Sopenharmony_ci if (ntfs_attr_pwrite(na, 0, ni->attr_list_size, 1007987da915Sopenharmony_ci ni->attr_list) != ni->attr_list_size) { 1008987da915Sopenharmony_ci if (!err || errno == EIO) { 1009987da915Sopenharmony_ci err = errno; 1010987da915Sopenharmony_ci if (err != EIO) 1011987da915Sopenharmony_ci err = EBUSY; 1012987da915Sopenharmony_ci ntfs_log_perror("Attribute list sync " 1013987da915Sopenharmony_ci "failed (write, inode %lld)", 1014987da915Sopenharmony_ci (long long)ni->mft_no); 1015987da915Sopenharmony_ci } 1016987da915Sopenharmony_ci NInoAttrListSetDirty(ni); 1017987da915Sopenharmony_ci } 1018987da915Sopenharmony_ci } else { 1019987da915Sopenharmony_ci err = EIO; 1020987da915Sopenharmony_ci ntfs_log_error("Attribute list sync failed (bad size, " 1021987da915Sopenharmony_ci "inode %lld)\n", (long long)ni->mft_no); 1022987da915Sopenharmony_ci NInoAttrListSetDirty(ni); 1023987da915Sopenharmony_ci } 1024987da915Sopenharmony_ci ntfs_attr_close(na); 1025987da915Sopenharmony_ci } 1026987da915Sopenharmony_ci 1027987da915Sopenharmony_cisync_inode: 1028987da915Sopenharmony_ci /* Write this inode out to the $MFT (and $MFTMirr if applicable). */ 1029987da915Sopenharmony_ci if (NInoTestAndClearDirty(ni)) { 1030987da915Sopenharmony_ci if (ntfs_mft_record_write(ni->vol, ni->mft_no, ni->mrec)) { 1031987da915Sopenharmony_ci if (!err || errno == EIO) { 1032987da915Sopenharmony_ci err = errno; 1033987da915Sopenharmony_ci if (err != EIO) 1034987da915Sopenharmony_ci err = EBUSY; 1035987da915Sopenharmony_ci } 1036987da915Sopenharmony_ci NInoSetDirty(ni); 1037987da915Sopenharmony_ci ntfs_log_perror("MFT record sync failed, inode %lld", 1038987da915Sopenharmony_ci (long long)ni->mft_no); 1039987da915Sopenharmony_ci } 1040987da915Sopenharmony_ci } 1041987da915Sopenharmony_ci 1042987da915Sopenharmony_ci /* If this is a base inode with extents write all dirty extents, too. */ 1043987da915Sopenharmony_ci if (ni->nr_extents > 0) { 1044987da915Sopenharmony_ci s32 i; 1045987da915Sopenharmony_ci 1046987da915Sopenharmony_ci for (i = 0; i < ni->nr_extents; ++i) { 1047987da915Sopenharmony_ci ntfs_inode *eni; 1048987da915Sopenharmony_ci 1049987da915Sopenharmony_ci eni = ni->extent_nis[i]; 1050987da915Sopenharmony_ci if (!NInoTestAndClearDirty(eni)) 1051987da915Sopenharmony_ci continue; 1052987da915Sopenharmony_ci 1053987da915Sopenharmony_ci if (ntfs_mft_record_write(eni->vol, eni->mft_no, 1054987da915Sopenharmony_ci eni->mrec)) { 1055987da915Sopenharmony_ci if (!err || errno == EIO) { 1056987da915Sopenharmony_ci err = errno; 1057987da915Sopenharmony_ci if (err != EIO) 1058987da915Sopenharmony_ci err = EBUSY; 1059987da915Sopenharmony_ci } 1060987da915Sopenharmony_ci NInoSetDirty(eni); 1061987da915Sopenharmony_ci ntfs_log_perror("Extent MFT record sync failed," 1062987da915Sopenharmony_ci " inode %lld/%lld", 1063987da915Sopenharmony_ci (long long)ni->mft_no, 1064987da915Sopenharmony_ci (long long)eni->mft_no); 1065987da915Sopenharmony_ci } 1066987da915Sopenharmony_ci } 1067987da915Sopenharmony_ci } 1068987da915Sopenharmony_ci 1069987da915Sopenharmony_ci if (err) { 1070987da915Sopenharmony_ci errno = err; 1071987da915Sopenharmony_ci ret = -1; 1072987da915Sopenharmony_ci } 1073987da915Sopenharmony_ci 1074987da915Sopenharmony_ci ntfs_log_leave("\n"); 1075987da915Sopenharmony_ci return ret; 1076987da915Sopenharmony_ci} 1077987da915Sopenharmony_ci 1078987da915Sopenharmony_ciint ntfs_inode_sync(ntfs_inode *ni) 1079987da915Sopenharmony_ci{ 1080987da915Sopenharmony_ci return (ntfs_inode_sync_in_dir(ni, (ntfs_inode*)NULL)); 1081987da915Sopenharmony_ci} 1082987da915Sopenharmony_ci 1083987da915Sopenharmony_ci/* 1084987da915Sopenharmony_ci * Close an inode with an open parent inode 1085987da915Sopenharmony_ci */ 1086987da915Sopenharmony_ci 1087987da915Sopenharmony_ciint ntfs_inode_close_in_dir(ntfs_inode *ni, ntfs_inode *dir_ni) 1088987da915Sopenharmony_ci{ 1089987da915Sopenharmony_ci int res; 1090987da915Sopenharmony_ci 1091987da915Sopenharmony_ci res = ntfs_inode_sync_in_dir(ni, dir_ni); 1092987da915Sopenharmony_ci if (res) { 1093987da915Sopenharmony_ci if (errno != EIO) 1094987da915Sopenharmony_ci errno = EBUSY; 1095987da915Sopenharmony_ci } else 1096987da915Sopenharmony_ci res = ntfs_inode_close(ni); 1097987da915Sopenharmony_ci return (res); 1098987da915Sopenharmony_ci} 1099987da915Sopenharmony_ci 1100987da915Sopenharmony_ci/** 1101987da915Sopenharmony_ci * ntfs_inode_add_attrlist - add attribute list to inode and fill it 1102987da915Sopenharmony_ci * @ni: opened ntfs inode to which add attribute list 1103987da915Sopenharmony_ci * 1104987da915Sopenharmony_ci * Return 0 on success or -1 on error with errno set to the error code. 1105987da915Sopenharmony_ci * The following error codes are defined: 1106987da915Sopenharmony_ci * EINVAL - Invalid arguments were passed to the function. 1107987da915Sopenharmony_ci * EEXIST - Attribute list already exist. 1108987da915Sopenharmony_ci * EIO - Input/Ouput error occurred. 1109987da915Sopenharmony_ci * ENOMEM - Not enough memory to perform add. 1110987da915Sopenharmony_ci */ 1111987da915Sopenharmony_ciint ntfs_inode_add_attrlist(ntfs_inode *ni) 1112987da915Sopenharmony_ci{ 1113987da915Sopenharmony_ci int err; 1114987da915Sopenharmony_ci ntfs_attr_search_ctx *ctx; 1115987da915Sopenharmony_ci u8 *al = NULL, *aln; 1116987da915Sopenharmony_ci int al_len = 0; 1117987da915Sopenharmony_ci ATTR_LIST_ENTRY *ale = NULL; 1118987da915Sopenharmony_ci ntfs_attr *na; 1119987da915Sopenharmony_ci 1120987da915Sopenharmony_ci if (!ni) { 1121987da915Sopenharmony_ci errno = EINVAL; 1122987da915Sopenharmony_ci ntfs_log_perror("%s", __FUNCTION__); 1123987da915Sopenharmony_ci return -1; 1124987da915Sopenharmony_ci } 1125987da915Sopenharmony_ci 1126987da915Sopenharmony_ci ntfs_log_trace("inode %llu\n", (unsigned long long) ni->mft_no); 1127987da915Sopenharmony_ci 1128987da915Sopenharmony_ci if (NInoAttrList(ni) || ni->nr_extents) { 1129987da915Sopenharmony_ci errno = EEXIST; 1130987da915Sopenharmony_ci ntfs_log_perror("Inode already has attribute list"); 1131987da915Sopenharmony_ci return -1; 1132987da915Sopenharmony_ci } 1133987da915Sopenharmony_ci 1134987da915Sopenharmony_ci /* Form attribute list. */ 1135987da915Sopenharmony_ci ctx = ntfs_attr_get_search_ctx(ni, NULL); 1136987da915Sopenharmony_ci if (!ctx) { 1137987da915Sopenharmony_ci err = errno; 1138987da915Sopenharmony_ci goto err_out; 1139987da915Sopenharmony_ci } 1140987da915Sopenharmony_ci /* Walk through all attributes. */ 1141987da915Sopenharmony_ci while (!ntfs_attr_lookup(AT_UNUSED, NULL, 0, 0, 0, NULL, 0, ctx)) { 1142987da915Sopenharmony_ci 1143987da915Sopenharmony_ci int ale_size; 1144987da915Sopenharmony_ci 1145987da915Sopenharmony_ci if (ctx->attr->type == AT_ATTRIBUTE_LIST) { 1146987da915Sopenharmony_ci err = EIO; 1147987da915Sopenharmony_ci ntfs_log_perror("Attribute list already present"); 1148987da915Sopenharmony_ci goto put_err_out; 1149987da915Sopenharmony_ci } 1150987da915Sopenharmony_ci 1151987da915Sopenharmony_ci ale_size = (sizeof(ATTR_LIST_ENTRY) + sizeof(ntfschar) * 1152987da915Sopenharmony_ci ctx->attr->name_length + 7) & ~7; 1153987da915Sopenharmony_ci al_len += ale_size; 1154987da915Sopenharmony_ci 1155987da915Sopenharmony_ci aln = realloc(al, al_len); 1156987da915Sopenharmony_ci if (!aln) { 1157987da915Sopenharmony_ci err = errno; 1158987da915Sopenharmony_ci ntfs_log_perror("Failed to realloc %d bytes", al_len); 1159987da915Sopenharmony_ci goto put_err_out; 1160987da915Sopenharmony_ci } 1161987da915Sopenharmony_ci ale = (ATTR_LIST_ENTRY *)(aln + ((u8 *)ale - al)); 1162987da915Sopenharmony_ci al = aln; 1163987da915Sopenharmony_ci 1164987da915Sopenharmony_ci memset(ale, 0, ale_size); 1165987da915Sopenharmony_ci 1166987da915Sopenharmony_ci /* Add attribute to attribute list. */ 1167987da915Sopenharmony_ci ale->type = ctx->attr->type; 1168987da915Sopenharmony_ci ale->length = cpu_to_le16((sizeof(ATTR_LIST_ENTRY) + 1169987da915Sopenharmony_ci sizeof(ntfschar) * ctx->attr->name_length + 7) & ~7); 1170987da915Sopenharmony_ci ale->name_length = ctx->attr->name_length; 1171987da915Sopenharmony_ci ale->name_offset = (u8 *)ale->name - (u8 *)ale; 1172987da915Sopenharmony_ci if (ctx->attr->non_resident) 1173987da915Sopenharmony_ci ale->lowest_vcn = ctx->attr->lowest_vcn; 1174987da915Sopenharmony_ci else 1175987da915Sopenharmony_ci ale->lowest_vcn = const_cpu_to_sle64(0); 1176987da915Sopenharmony_ci ale->mft_reference = MK_LE_MREF(ni->mft_no, 1177987da915Sopenharmony_ci le16_to_cpu(ni->mrec->sequence_number)); 1178987da915Sopenharmony_ci ale->instance = ctx->attr->instance; 1179987da915Sopenharmony_ci memcpy(ale->name, (u8 *)ctx->attr + 1180987da915Sopenharmony_ci le16_to_cpu(ctx->attr->name_offset), 1181987da915Sopenharmony_ci ctx->attr->name_length * sizeof(ntfschar)); 1182987da915Sopenharmony_ci ale = (ATTR_LIST_ENTRY *)(al + al_len); 1183987da915Sopenharmony_ci } 1184987da915Sopenharmony_ci /* Check for real error occurred. */ 1185987da915Sopenharmony_ci if (errno != ENOENT) { 1186987da915Sopenharmony_ci err = errno; 1187987da915Sopenharmony_ci ntfs_log_perror("%s: Attribute lookup failed, inode %lld", 1188987da915Sopenharmony_ci __FUNCTION__, (long long)ni->mft_no); 1189987da915Sopenharmony_ci goto put_err_out; 1190987da915Sopenharmony_ci } 1191987da915Sopenharmony_ci 1192987da915Sopenharmony_ci /* Set in-memory attribute list. */ 1193987da915Sopenharmony_ci ni->attr_list = al; 1194987da915Sopenharmony_ci ni->attr_list_size = al_len; 1195987da915Sopenharmony_ci NInoSetAttrList(ni); 1196987da915Sopenharmony_ci NInoAttrListSetDirty(ni); 1197987da915Sopenharmony_ci 1198987da915Sopenharmony_ci /* Free space if there is not enough it for $ATTRIBUTE_LIST. */ 1199987da915Sopenharmony_ci if (le32_to_cpu(ni->mrec->bytes_allocated) - 1200987da915Sopenharmony_ci le32_to_cpu(ni->mrec->bytes_in_use) < 1201987da915Sopenharmony_ci offsetof(ATTR_RECORD, resident_end)) { 1202987da915Sopenharmony_ci if (ntfs_inode_free_space(ni, 1203987da915Sopenharmony_ci offsetof(ATTR_RECORD, resident_end))) { 1204987da915Sopenharmony_ci /* Failed to free space. */ 1205987da915Sopenharmony_ci err = errno; 1206987da915Sopenharmony_ci ntfs_log_perror("Failed to free space for attrlist"); 1207987da915Sopenharmony_ci goto rollback; 1208987da915Sopenharmony_ci } 1209987da915Sopenharmony_ci } 1210987da915Sopenharmony_ci 1211987da915Sopenharmony_ci /* Add $ATTRIBUTE_LIST to mft record. */ 1212987da915Sopenharmony_ci if (ntfs_resident_attr_record_add(ni, 1213987da915Sopenharmony_ci AT_ATTRIBUTE_LIST, NULL, 0, NULL, 0, const_cpu_to_le16(0)) < 0) { 1214987da915Sopenharmony_ci err = errno; 1215987da915Sopenharmony_ci ntfs_log_perror("Couldn't add $ATTRIBUTE_LIST to MFT"); 1216987da915Sopenharmony_ci goto rollback; 1217987da915Sopenharmony_ci } 1218987da915Sopenharmony_ci 1219987da915Sopenharmony_ci /* Resize it. */ 1220987da915Sopenharmony_ci na = ntfs_attr_open(ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0); 1221987da915Sopenharmony_ci if (!na) { 1222987da915Sopenharmony_ci err = errno; 1223987da915Sopenharmony_ci ntfs_log_perror("Failed to open just added $ATTRIBUTE_LIST"); 1224987da915Sopenharmony_ci goto remove_attrlist_record; 1225987da915Sopenharmony_ci } 1226987da915Sopenharmony_ci if (ntfs_attr_truncate(na, al_len)) { 1227987da915Sopenharmony_ci err = errno; 1228987da915Sopenharmony_ci ntfs_log_perror("Failed to resize just added $ATTRIBUTE_LIST"); 1229987da915Sopenharmony_ci ntfs_attr_close(na); 1230987da915Sopenharmony_ci goto remove_attrlist_record;; 1231987da915Sopenharmony_ci } 1232987da915Sopenharmony_ci 1233987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 1234987da915Sopenharmony_ci ntfs_attr_close(na); 1235987da915Sopenharmony_ci return 0; 1236987da915Sopenharmony_ci 1237987da915Sopenharmony_ciremove_attrlist_record: 1238987da915Sopenharmony_ci /* Prevent ntfs_attr_recorm_rm from freeing attribute list. */ 1239987da915Sopenharmony_ci ni->attr_list = NULL; 1240987da915Sopenharmony_ci NInoClearAttrList(ni); 1241987da915Sopenharmony_ci /* Remove $ATTRIBUTE_LIST record. */ 1242987da915Sopenharmony_ci ntfs_attr_reinit_search_ctx(ctx); 1243987da915Sopenharmony_ci if (!ntfs_attr_lookup(AT_ATTRIBUTE_LIST, NULL, 0, 1244987da915Sopenharmony_ci CASE_SENSITIVE, 0, NULL, 0, ctx)) { 1245987da915Sopenharmony_ci if (ntfs_attr_record_rm(ctx)) 1246987da915Sopenharmony_ci ntfs_log_perror("Rollback failed to remove attrlist"); 1247987da915Sopenharmony_ci } else 1248987da915Sopenharmony_ci ntfs_log_perror("Rollback failed to find attrlist"); 1249987da915Sopenharmony_ci /* Setup back in-memory runlist. */ 1250987da915Sopenharmony_ci ni->attr_list = al; 1251987da915Sopenharmony_ci ni->attr_list_size = al_len; 1252987da915Sopenharmony_ci NInoSetAttrList(ni); 1253987da915Sopenharmony_cirollback: 1254987da915Sopenharmony_ci /* 1255987da915Sopenharmony_ci * Scan attribute list for attributes that placed not in the base MFT 1256987da915Sopenharmony_ci * record and move them to it. 1257987da915Sopenharmony_ci */ 1258987da915Sopenharmony_ci ntfs_attr_reinit_search_ctx(ctx); 1259987da915Sopenharmony_ci ale = (ATTR_LIST_ENTRY*)al; 1260987da915Sopenharmony_ci while ((u8*)ale < al + al_len) { 1261987da915Sopenharmony_ci if (MREF_LE(ale->mft_reference) != ni->mft_no) { 1262987da915Sopenharmony_ci if (!ntfs_attr_lookup(ale->type, ale->name, 1263987da915Sopenharmony_ci ale->name_length, 1264987da915Sopenharmony_ci CASE_SENSITIVE, 1265987da915Sopenharmony_ci sle64_to_cpu(ale->lowest_vcn), 1266987da915Sopenharmony_ci NULL, 0, ctx)) { 1267987da915Sopenharmony_ci if (ntfs_attr_record_move_to(ctx, ni)) 1268987da915Sopenharmony_ci ntfs_log_perror("Rollback failed to " 1269987da915Sopenharmony_ci "move attribute"); 1270987da915Sopenharmony_ci } else 1271987da915Sopenharmony_ci ntfs_log_perror("Rollback failed to find attr"); 1272987da915Sopenharmony_ci ntfs_attr_reinit_search_ctx(ctx); 1273987da915Sopenharmony_ci } 1274987da915Sopenharmony_ci ale = (ATTR_LIST_ENTRY*)((u8*)ale + le16_to_cpu(ale->length)); 1275987da915Sopenharmony_ci } 1276987da915Sopenharmony_ci /* Remove in-memory attribute list. */ 1277987da915Sopenharmony_ci ni->attr_list = NULL; 1278987da915Sopenharmony_ci ni->attr_list_size = 0; 1279987da915Sopenharmony_ci NInoClearAttrList(ni); 1280987da915Sopenharmony_ci NInoAttrListClearDirty(ni); 1281987da915Sopenharmony_ciput_err_out: 1282987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 1283987da915Sopenharmony_cierr_out: 1284987da915Sopenharmony_ci free(al); 1285987da915Sopenharmony_ci errno = err; 1286987da915Sopenharmony_ci return -1; 1287987da915Sopenharmony_ci} 1288987da915Sopenharmony_ci 1289987da915Sopenharmony_ci/** 1290987da915Sopenharmony_ci * ntfs_inode_free_space - free space in the MFT record of an inode 1291987da915Sopenharmony_ci * @ni: ntfs inode in which MFT record needs more free space 1292987da915Sopenharmony_ci * @size: amount of space needed to free 1293987da915Sopenharmony_ci * 1294987da915Sopenharmony_ci * Return 0 on success or -1 on error with errno set to the error code. 1295987da915Sopenharmony_ci */ 1296987da915Sopenharmony_ciint ntfs_inode_free_space(ntfs_inode *ni, int size) 1297987da915Sopenharmony_ci{ 1298987da915Sopenharmony_ci ntfs_attr_search_ctx *ctx; 1299987da915Sopenharmony_ci int freed; 1300987da915Sopenharmony_ci 1301987da915Sopenharmony_ci if (!ni || size < 0) { 1302987da915Sopenharmony_ci errno = EINVAL; 1303987da915Sopenharmony_ci ntfs_log_perror("%s: ni=%p size=%d", __FUNCTION__, ni, size); 1304987da915Sopenharmony_ci return -1; 1305987da915Sopenharmony_ci } 1306987da915Sopenharmony_ci 1307987da915Sopenharmony_ci ntfs_log_trace("Entering for inode %lld, size %d\n", 1308987da915Sopenharmony_ci (unsigned long long)ni->mft_no, size); 1309987da915Sopenharmony_ci 1310987da915Sopenharmony_ci freed = (le32_to_cpu(ni->mrec->bytes_allocated) - 1311987da915Sopenharmony_ci le32_to_cpu(ni->mrec->bytes_in_use)); 1312987da915Sopenharmony_ci 1313987da915Sopenharmony_ci if (size <= freed) 1314987da915Sopenharmony_ci return 0; 1315987da915Sopenharmony_ci 1316987da915Sopenharmony_ci ctx = ntfs_attr_get_search_ctx(ni, NULL); 1317987da915Sopenharmony_ci if (!ctx) 1318987da915Sopenharmony_ci return -1; 1319987da915Sopenharmony_ci /* 1320987da915Sopenharmony_ci * $STANDARD_INFORMATION and $ATTRIBUTE_LIST must stay in the base MFT 1321987da915Sopenharmony_ci * record, so position search context on the first attribute after them. 1322987da915Sopenharmony_ci */ 1323987da915Sopenharmony_ci if (ntfs_attr_position(AT_FILE_NAME, ctx)) 1324987da915Sopenharmony_ci goto put_err_out; 1325987da915Sopenharmony_ci 1326987da915Sopenharmony_ci while (1) { 1327987da915Sopenharmony_ci int record_size; 1328987da915Sopenharmony_ci /* 1329987da915Sopenharmony_ci * Check whether attribute is from different MFT record. If so, 1330987da915Sopenharmony_ci * find next, because we don't need such. 1331987da915Sopenharmony_ci */ 1332987da915Sopenharmony_ci while (ctx->ntfs_ino->mft_no != ni->mft_no) { 1333987da915Sopenharmony_ciretry: 1334987da915Sopenharmony_ci if (ntfs_attr_position(AT_UNUSED, ctx)) 1335987da915Sopenharmony_ci goto put_err_out; 1336987da915Sopenharmony_ci } 1337987da915Sopenharmony_ci 1338987da915Sopenharmony_ci if (ntfs_inode_base(ctx->ntfs_ino)->mft_no == FILE_MFT && 1339987da915Sopenharmony_ci ctx->attr->type == AT_DATA) 1340987da915Sopenharmony_ci goto retry; 1341987da915Sopenharmony_ci 1342987da915Sopenharmony_ci if (ctx->attr->type == AT_INDEX_ROOT) 1343987da915Sopenharmony_ci goto retry; 1344987da915Sopenharmony_ci 1345987da915Sopenharmony_ci record_size = le32_to_cpu(ctx->attr->length); 1346987da915Sopenharmony_ci 1347987da915Sopenharmony_ci if (ntfs_attr_record_move_away(ctx, 0)) { 1348987da915Sopenharmony_ci ntfs_log_perror("Failed to move out attribute #2"); 1349987da915Sopenharmony_ci break; 1350987da915Sopenharmony_ci } 1351987da915Sopenharmony_ci freed += record_size; 1352987da915Sopenharmony_ci 1353987da915Sopenharmony_ci /* Check whether we are done. */ 1354987da915Sopenharmony_ci if (size <= freed) { 1355987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 1356987da915Sopenharmony_ci return 0; 1357987da915Sopenharmony_ci } 1358987da915Sopenharmony_ci /* 1359987da915Sopenharmony_ci * Reposition to first attribute after $STANDARD_INFORMATION 1360987da915Sopenharmony_ci * and $ATTRIBUTE_LIST instead of simply skipping this attribute 1361987da915Sopenharmony_ci * because in the case when we have got only in-memory attribute 1362987da915Sopenharmony_ci * list then ntfs_attr_lookup will fail when it tries to find 1363987da915Sopenharmony_ci * $ATTRIBUTE_LIST. 1364987da915Sopenharmony_ci */ 1365987da915Sopenharmony_ci ntfs_attr_reinit_search_ctx(ctx); 1366987da915Sopenharmony_ci if (ntfs_attr_position(AT_FILE_NAME, ctx)) 1367987da915Sopenharmony_ci break; 1368987da915Sopenharmony_ci } 1369987da915Sopenharmony_ciput_err_out: 1370987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 1371987da915Sopenharmony_ci if (errno == ENOSPC) 1372987da915Sopenharmony_ci ntfs_log_trace("No attributes left that could be moved out.\n"); 1373987da915Sopenharmony_ci return -1; 1374987da915Sopenharmony_ci} 1375987da915Sopenharmony_ci 1376987da915Sopenharmony_ci/** 1377987da915Sopenharmony_ci * ntfs_inode_update_times - update selected time fields for ntfs inode 1378987da915Sopenharmony_ci * @ni: ntfs inode for which update time fields 1379987da915Sopenharmony_ci * @mask: select which time fields should be updated 1380987da915Sopenharmony_ci * 1381987da915Sopenharmony_ci * This function updates time fields to current time. Fields to update are 1382987da915Sopenharmony_ci * selected using @mask (see enum @ntfs_time_update_flags for posssible values). 1383987da915Sopenharmony_ci */ 1384987da915Sopenharmony_civoid ntfs_inode_update_times(ntfs_inode *ni, ntfs_time_update_flags mask) 1385987da915Sopenharmony_ci{ 1386987da915Sopenharmony_ci ntfs_time now; 1387987da915Sopenharmony_ci 1388987da915Sopenharmony_ci if (!ni) { 1389987da915Sopenharmony_ci ntfs_log_error("%s(): Invalid arguments.\n", __FUNCTION__); 1390987da915Sopenharmony_ci return; 1391987da915Sopenharmony_ci } 1392987da915Sopenharmony_ci 1393987da915Sopenharmony_ci if ((ni->mft_no < FILE_first_user && ni->mft_no != FILE_root) || 1394987da915Sopenharmony_ci NVolReadOnly(ni->vol) || !mask) 1395987da915Sopenharmony_ci return; 1396987da915Sopenharmony_ci 1397987da915Sopenharmony_ci now = ntfs_current_time(); 1398987da915Sopenharmony_ci if (mask & NTFS_UPDATE_ATIME) 1399987da915Sopenharmony_ci ni->last_access_time = now; 1400987da915Sopenharmony_ci if (mask & NTFS_UPDATE_MTIME) 1401987da915Sopenharmony_ci ni->last_data_change_time = now; 1402987da915Sopenharmony_ci if (mask & NTFS_UPDATE_CTIME) 1403987da915Sopenharmony_ci ni->last_mft_change_time = now; 1404987da915Sopenharmony_ci 1405987da915Sopenharmony_ci NInoFileNameSetDirty(ni); 1406987da915Sopenharmony_ci NInoSetDirty(ni); 1407987da915Sopenharmony_ci} 1408987da915Sopenharmony_ci 1409987da915Sopenharmony_ci/** 1410987da915Sopenharmony_ci * ntfs_inode_badclus_bad - check for $Badclus:$Bad data attribute 1411987da915Sopenharmony_ci * @mft_no: mft record number where @attr is present 1412987da915Sopenharmony_ci * @attr: attribute record used to check for the $Bad attribute 1413987da915Sopenharmony_ci * 1414987da915Sopenharmony_ci * Check if the mft record given by @mft_no and @attr contains the bad sector 1415987da915Sopenharmony_ci * list. Please note that mft record numbers describing $Badclus extent inodes 1416987da915Sopenharmony_ci * will not match the current $Badclus:$Bad check. 1417987da915Sopenharmony_ci * 1418987da915Sopenharmony_ci * On success return 1 if the file is $Badclus:$Bad, otherwise return 0. 1419987da915Sopenharmony_ci * On error return -1 with errno set to the error code. 1420987da915Sopenharmony_ci */ 1421987da915Sopenharmony_ciint ntfs_inode_badclus_bad(u64 mft_no, ATTR_RECORD *attr) 1422987da915Sopenharmony_ci{ 1423987da915Sopenharmony_ci int len, ret = 0; 1424987da915Sopenharmony_ci ntfschar *ustr; 1425987da915Sopenharmony_ci 1426987da915Sopenharmony_ci if (!attr) { 1427987da915Sopenharmony_ci ntfs_log_error("Invalid argument.\n"); 1428987da915Sopenharmony_ci errno = EINVAL; 1429987da915Sopenharmony_ci return -1; 1430987da915Sopenharmony_ci } 1431987da915Sopenharmony_ci 1432987da915Sopenharmony_ci if (mft_no != FILE_BadClus) 1433987da915Sopenharmony_ci return 0; 1434987da915Sopenharmony_ci 1435987da915Sopenharmony_ci if (attr->type != AT_DATA) 1436987da915Sopenharmony_ci return 0; 1437987da915Sopenharmony_ci 1438987da915Sopenharmony_ci if ((ustr = ntfs_str2ucs("$Bad", &len)) == NULL) { 1439987da915Sopenharmony_ci ntfs_log_perror("Couldn't convert '$Bad' to Unicode"); 1440987da915Sopenharmony_ci return -1; 1441987da915Sopenharmony_ci } 1442987da915Sopenharmony_ci 1443987da915Sopenharmony_ci if (ustr && ntfs_names_are_equal(ustr, len, 1444987da915Sopenharmony_ci (ntfschar *)((u8 *)attr + le16_to_cpu(attr->name_offset)), 1445987da915Sopenharmony_ci attr->name_length, 0, NULL, 0)) 1446987da915Sopenharmony_ci ret = 1; 1447987da915Sopenharmony_ci 1448987da915Sopenharmony_ci ntfs_ucsfree(ustr); 1449987da915Sopenharmony_ci 1450987da915Sopenharmony_ci return ret; 1451987da915Sopenharmony_ci} 1452987da915Sopenharmony_ci 1453987da915Sopenharmony_ci/* 1454987da915Sopenharmony_ci * Get high precision NTFS times 1455987da915Sopenharmony_ci * 1456987da915Sopenharmony_ci * They are returned in following order : create, update, access, change 1457987da915Sopenharmony_ci * provided they fit in requested size. 1458987da915Sopenharmony_ci * 1459987da915Sopenharmony_ci * Returns the modified size if successfull (or 32 if buffer size is null) 1460987da915Sopenharmony_ci * -errno if failed 1461987da915Sopenharmony_ci */ 1462987da915Sopenharmony_ci 1463987da915Sopenharmony_ciint ntfs_inode_get_times(ntfs_inode *ni, char *value, size_t size) 1464987da915Sopenharmony_ci{ 1465987da915Sopenharmony_ci ntfs_attr_search_ctx *ctx; 1466987da915Sopenharmony_ci STANDARD_INFORMATION *std_info; 1467987da915Sopenharmony_ci u64 *times; 1468987da915Sopenharmony_ci int ret; 1469987da915Sopenharmony_ci 1470987da915Sopenharmony_ci ret = 0; 1471987da915Sopenharmony_ci ctx = ntfs_attr_get_search_ctx(ni, NULL); 1472987da915Sopenharmony_ci if (ctx) { 1473987da915Sopenharmony_ci if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, AT_UNNAMED, 1474987da915Sopenharmony_ci 0, CASE_SENSITIVE, 0, NULL, 0, ctx)) { 1475987da915Sopenharmony_ci ntfs_log_perror("Failed to get standard info (inode %lld)", 1476987da915Sopenharmony_ci (long long)ni->mft_no); 1477987da915Sopenharmony_ci } else { 1478987da915Sopenharmony_ci std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + 1479987da915Sopenharmony_ci le16_to_cpu(ctx->attr->value_offset)); 1480987da915Sopenharmony_ci if (value && (size >= 8)) { 1481987da915Sopenharmony_ci times = (u64*)value; 1482987da915Sopenharmony_ci times[0] = sle64_to_cpu(std_info->creation_time); 1483987da915Sopenharmony_ci ret = 8; 1484987da915Sopenharmony_ci if (size >= 16) { 1485987da915Sopenharmony_ci times[1] = sle64_to_cpu(std_info->last_data_change_time); 1486987da915Sopenharmony_ci ret = 16; 1487987da915Sopenharmony_ci } 1488987da915Sopenharmony_ci if (size >= 24) { 1489987da915Sopenharmony_ci times[2] = sle64_to_cpu(std_info->last_access_time); 1490987da915Sopenharmony_ci ret = 24; 1491987da915Sopenharmony_ci } 1492987da915Sopenharmony_ci if (size >= 32) { 1493987da915Sopenharmony_ci times[3] = sle64_to_cpu(std_info->last_mft_change_time); 1494987da915Sopenharmony_ci ret = 32; 1495987da915Sopenharmony_ci } 1496987da915Sopenharmony_ci } else 1497987da915Sopenharmony_ci if (!size) 1498987da915Sopenharmony_ci ret = 32; 1499987da915Sopenharmony_ci else 1500987da915Sopenharmony_ci ret = -ERANGE; 1501987da915Sopenharmony_ci } 1502987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 1503987da915Sopenharmony_ci } 1504987da915Sopenharmony_ci return (ret ? ret : -errno); 1505987da915Sopenharmony_ci} 1506987da915Sopenharmony_ci 1507987da915Sopenharmony_ci/* 1508987da915Sopenharmony_ci * Set high precision NTFS times 1509987da915Sopenharmony_ci * 1510987da915Sopenharmony_ci * They are expected in this order : create, update, access 1511987da915Sopenharmony_ci * provided they are present in input. The change time is set to 1512987da915Sopenharmony_ci * current time. 1513987da915Sopenharmony_ci * 1514987da915Sopenharmony_ci * The times are inserted directly in the standard_information and 1515987da915Sopenharmony_ci * file names attributes to avoid manipulating low precision times 1516987da915Sopenharmony_ci * 1517987da915Sopenharmony_ci * Returns 0 if success 1518987da915Sopenharmony_ci * -1 if there were an error (described by errno) 1519987da915Sopenharmony_ci */ 1520987da915Sopenharmony_ci 1521987da915Sopenharmony_ciint ntfs_inode_set_times(ntfs_inode *ni, const char *value, size_t size, 1522987da915Sopenharmony_ci int flags) 1523987da915Sopenharmony_ci{ 1524987da915Sopenharmony_ci ntfs_attr_search_ctx *ctx; 1525987da915Sopenharmony_ci STANDARD_INFORMATION *std_info; 1526987da915Sopenharmony_ci FILE_NAME_ATTR *fn; 1527987da915Sopenharmony_ci u64 times[4]; 1528987da915Sopenharmony_ci ntfs_time now; 1529987da915Sopenharmony_ci int cnt; 1530987da915Sopenharmony_ci int ret; 1531987da915Sopenharmony_ci 1532987da915Sopenharmony_ci ret = -1; 1533987da915Sopenharmony_ci if ((size >= 8) && !(flags & XATTR_CREATE)) { 1534987da915Sopenharmony_ci /* Copy, to avoid alignment issue encountered on ARM */ 1535987da915Sopenharmony_ci memcpy(times, value, 1536987da915Sopenharmony_ci (size < sizeof(times) ? size : sizeof(times))); 1537987da915Sopenharmony_ci now = ntfs_current_time(); 1538987da915Sopenharmony_ci /* update the standard information attribute */ 1539987da915Sopenharmony_ci ctx = ntfs_attr_get_search_ctx(ni, NULL); 1540987da915Sopenharmony_ci if (ctx) { 1541987da915Sopenharmony_ci if (ntfs_attr_lookup(AT_STANDARD_INFORMATION, 1542987da915Sopenharmony_ci AT_UNNAMED, 0, CASE_SENSITIVE, 1543987da915Sopenharmony_ci 0, NULL, 0, ctx)) { 1544987da915Sopenharmony_ci ntfs_log_perror("Failed to get standard info (inode %lld)", 1545987da915Sopenharmony_ci (long long)ni->mft_no); 1546987da915Sopenharmony_ci } else { 1547987da915Sopenharmony_ci std_info = (STANDARD_INFORMATION *)((u8 *)ctx->attr + 1548987da915Sopenharmony_ci le16_to_cpu(ctx->attr->value_offset)); 1549987da915Sopenharmony_ci /* 1550987da915Sopenharmony_ci * Mark times set to avoid overwriting 1551987da915Sopenharmony_ci * them when the inode is closed. 1552987da915Sopenharmony_ci * The inode structure must also be updated 1553987da915Sopenharmony_ci * (with loss of precision) because of cacheing. 1554987da915Sopenharmony_ci * TODO : use NTFS precision in inode, and 1555987da915Sopenharmony_ci * return sub-second times in getattr() 1556987da915Sopenharmony_ci */ 1557987da915Sopenharmony_ci set_nino_flag(ni, TimesSet); 1558987da915Sopenharmony_ci std_info->creation_time = cpu_to_sle64(times[0]); 1559987da915Sopenharmony_ci ni->creation_time 1560987da915Sopenharmony_ci = std_info->creation_time; 1561987da915Sopenharmony_ci if (size >= 16) { 1562987da915Sopenharmony_ci std_info->last_data_change_time = cpu_to_sle64(times[1]); 1563987da915Sopenharmony_ci ni->last_data_change_time 1564987da915Sopenharmony_ci = std_info->last_data_change_time; 1565987da915Sopenharmony_ci } 1566987da915Sopenharmony_ci if (size >= 24) { 1567987da915Sopenharmony_ci std_info->last_access_time = cpu_to_sle64(times[2]); 1568987da915Sopenharmony_ci ni->last_access_time 1569987da915Sopenharmony_ci = std_info->last_access_time; 1570987da915Sopenharmony_ci } 1571987da915Sopenharmony_ci std_info->last_mft_change_time = now; 1572987da915Sopenharmony_ci ni->last_mft_change_time = now; 1573987da915Sopenharmony_ci ntfs_inode_mark_dirty(ctx->ntfs_ino); 1574987da915Sopenharmony_ci NInoFileNameSetDirty(ni); 1575987da915Sopenharmony_ci 1576987da915Sopenharmony_ci /* update the file names attributes */ 1577987da915Sopenharmony_ci ntfs_attr_reinit_search_ctx(ctx); 1578987da915Sopenharmony_ci cnt = 0; 1579987da915Sopenharmony_ci while (!ntfs_attr_lookup(AT_FILE_NAME, 1580987da915Sopenharmony_ci AT_UNNAMED, 0, CASE_SENSITIVE, 1581987da915Sopenharmony_ci 0, NULL, 0, ctx)) { 1582987da915Sopenharmony_ci fn = (FILE_NAME_ATTR*)((u8 *)ctx->attr + 1583987da915Sopenharmony_ci le16_to_cpu(ctx->attr->value_offset)); 1584987da915Sopenharmony_ci fn->creation_time 1585987da915Sopenharmony_ci = cpu_to_sle64(times[0]); 1586987da915Sopenharmony_ci if (size >= 16) 1587987da915Sopenharmony_ci fn->last_data_change_time 1588987da915Sopenharmony_ci = cpu_to_sle64(times[1]); 1589987da915Sopenharmony_ci if (size >= 24) 1590987da915Sopenharmony_ci fn->last_access_time 1591987da915Sopenharmony_ci = cpu_to_sle64(times[2]); 1592987da915Sopenharmony_ci fn->last_mft_change_time = now; 1593987da915Sopenharmony_ci cnt++; 1594987da915Sopenharmony_ci } 1595987da915Sopenharmony_ci if (cnt) 1596987da915Sopenharmony_ci ret = 0; 1597987da915Sopenharmony_ci else { 1598987da915Sopenharmony_ci ntfs_log_perror("Failed to get file names (inode %lld)", 1599987da915Sopenharmony_ci (long long)ni->mft_no); 1600987da915Sopenharmony_ci } 1601987da915Sopenharmony_ci } 1602987da915Sopenharmony_ci ntfs_attr_put_search_ctx(ctx); 1603987da915Sopenharmony_ci } 1604987da915Sopenharmony_ci } else 1605987da915Sopenharmony_ci if (size < 8) 1606987da915Sopenharmony_ci errno = ERANGE; 1607987da915Sopenharmony_ci else 1608987da915Sopenharmony_ci errno = EEXIST; 1609987da915Sopenharmony_ci return (ret); 1610987da915Sopenharmony_ci} 1611