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