summaryrefslogtreecommitdiff
Diffstat
-rwxr-xr-xAUTHORS23
-rwxr-xr-xAndroid.mk89
-rwxr-xr-xAndroid.mk.bak89
-rwxr-xr-xCOPYING340
-rwxr-xr-xCOPYING.LIB482
-rwxr-xr-xCREDITS41
-rwxr-xr-xChangeLog7
-rwxr-xr-xINSTALL237
-rwxr-xr-xMakefile715
-rwxr-xr-xMakefile.am66
-rwxr-xr-xMakefile.in722
-rwxr-xr-xNEWS5
-rwxr-xr-xREADME68
-rwxr-xr-xaclocal.m47723
-rwxr-xr-xautogen.sh22
-rwxr-xr-xcompile142
-rwxr-xr-xconfig.guess1536
-rwxr-xr-xconfig.h361
-rwxr-xr-xconfig.h.in350
-rwxr-xr-xconfig.log3290
-rwxr-xr-xconfig.status1220
-rwxr-xr-xconfig.sub1658
-rwxr-xr-xconfigure25302
-rwxr-xr-xconfigure.ac441
-rwxr-xr-xdepcomp589
-rwxr-xr-xinclude/Makefile.am4
-rwxr-xr-xinclude/Makefile.in503
-rwxr-xr-xinclude/fuse-lite/Makefile.am10
-rwxr-xr-xinclude/fuse-lite/Makefile.in400
-rwxr-xr-xinclude/fuse-lite/fuse.h654
-rwxr-xr-xinclude/fuse-lite/fuse_common.h193
-rwxr-xr-xinclude/fuse-lite/fuse_kernel.h422
-rwxr-xr-xinclude/fuse-lite/fuse_lowlevel.h1375
-rwxr-xr-xinclude/fuse-lite/fuse_lowlevel_compat.h16
-rwxr-xr-xinclude/fuse-lite/fuse_opt.h261
-rwxr-xr-xinclude/ntfs-3g/Makefile.am45
-rwxr-xr-xinclude/ntfs-3g/Makefile.in471
-rwxr-xr-xinclude/ntfs-3g/acls.h199
-rwxr-xr-xinclude/ntfs-3g/attrib.h358
-rwxr-xr-xinclude/ntfs-3g/attrlist.h51
-rwxr-xr-xinclude/ntfs-3g/bitmap.h96
-rwxr-xr-xinclude/ntfs-3g/bootsect.h42
-rwxr-xr-xinclude/ntfs-3g/cache.h115
-rwxr-xr-xinclude/ntfs-3g/collate.h34
-rwxr-xr-xinclude/ntfs-3g/compat.h69
-rwxr-xr-xinclude/ntfs-3g/compress.h39
-rwxr-xr-xinclude/ntfs-3g/debug.h47
-rwxr-xr-xinclude/ntfs-3g/device.h128
-rwxr-xr-xinclude/ntfs-3g/device_io.h77
-rwxr-xr-xinclude/ntfs-3g/dir.h128
-rwxr-xr-xinclude/ntfs-3g/efs.h30
-rwxr-xr-xinclude/ntfs-3g/endians.h203
-rwxr-xr-xinclude/ntfs-3g/index.h167
-rwxr-xr-xinclude/ntfs-3g/inode.h225
-rwxr-xr-xinclude/ntfs-3g/layout.h2661
-rwxr-xr-xinclude/ntfs-3g/lcnalloc.h50
-rwxr-xr-xinclude/ntfs-3g/logfile.h394
-rwxr-xr-xinclude/ntfs-3g/logging.h118
-rwxr-xr-xinclude/ntfs-3g/mft.h132
-rwxr-xr-xinclude/ntfs-3g/misc.h30
-rwxr-xr-xinclude/ntfs-3g/mst.h34
-rwxr-xr-xinclude/ntfs-3g/ntfstime.h131
-rwxr-xr-xinclude/ntfs-3g/object_id.h35
-rwxr-xr-xinclude/ntfs-3g/param.h63
-rwxr-xr-xinclude/ntfs-3g/reparse.h39
-rwxr-xr-xinclude/ntfs-3g/runlist.h89
-rwxr-xr-xinclude/ntfs-3g/security.h353
-rwxr-xr-xinclude/ntfs-3g/support.h85
-rwxr-xr-xinclude/ntfs-3g/types.h124
-rwxr-xr-xinclude/ntfs-3g/unistr.h116
-rwxr-xr-xinclude/ntfs-3g/volume.h277
-rwxr-xr-xinstall-sh519
-rwxr-xr-xlibfuse-lite/Makefile.am28
-rwxr-xr-xlibfuse-lite/Makefile.in572
-rwxr-xr-xlibfuse-lite/fuse.c2789
-rwxr-xr-xlibfuse-lite/fuse_i.h25
-rwxr-xr-xlibfuse-lite/fuse_kern_chan.c96
-rwxr-xr-xlibfuse-lite/fuse_loop.c40
-rwxr-xr-xlibfuse-lite/fuse_lowlevel.c1395
-rwxr-xr-xlibfuse-lite/fuse_misc.h106
-rwxr-xr-xlibfuse-lite/fuse_opt.c368
-rwxr-xr-xlibfuse-lite/fuse_session.c183
-rwxr-xr-xlibfuse-lite/fuse_signals.c73
-rwxr-xr-xlibfuse-lite/fusermount.c772
-rwxr-xr-xlibfuse-lite/helper.c40
-rwxr-xr-xlibfuse-lite/mount.c256
-rwxr-xr-xlibfuse-lite/mount_util.c219
-rwxr-xr-xlibfuse-lite/mount_util.h22
-rwxr-xr-xlibntfs-3g/Makefile.am79
-rwxr-xr-xlibntfs-3g/Makefile.in841
-rwxr-xr-xlibntfs-3g/acls.c4296
-rwxr-xr-xlibntfs-3g/attrib.c5913
-rwxr-xr-xlibntfs-3g/attrlist.c314
-rwxr-xr-xlibntfs-3g/bitmap.c300
-rwxr-xr-xlibntfs-3g/bootsect.c285
-rwxr-xr-xlibntfs-3g/cache.c609
-rwxr-xr-xlibntfs-3g/collate.c271
-rwxr-xr-xlibntfs-3g/compat.c250
-rwxr-xr-xlibntfs-3g/compress.c1431
-rwxr-xr-xlibntfs-3g/debug.c79
-rwxr-xr-xlibntfs-3g/device.c730
-rwxr-xr-xlibntfs-3g/dir.c2582
-rwxr-xr-xlibntfs-3g/efs.c346
-rwxr-xr-xlibntfs-3g/index.c2063
-rwxr-xr-xlibntfs-3g/inode.c1566
-rwxr-xr-xlibntfs-3g/lcnalloc.c735
-rwxr-xr-xlibntfs-3g/libntfs-3g.pc.in10
-rwxr-xr-xlibntfs-3g/libntfs-3g.script.so.in2
-rwxr-xr-xlibntfs-3g/logfile.c737
-rwxr-xr-xlibntfs-3g/logging.c613
-rwxr-xr-xlibntfs-3g/mft.c1909
-rwxr-xr-xlibntfs-3g/misc.c61
-rwxr-xr-xlibntfs-3g/mst.c231
-rwxr-xr-xlibntfs-3g/object_id.c637
-rwxr-xr-xlibntfs-3g/reparse.c1222
-rwxr-xr-xlibntfs-3g/runlist.c2166
-rwxr-xr-xlibntfs-3g/security.c5167
-rwxr-xr-xlibntfs-3g/unistr.c1321
-rwxr-xr-xlibntfs-3g/unix_io.c386
-rwxr-xr-xlibntfs-3g/volume.c1663
-rwxr-xr-xlibntfs-3g/win32_io.c1477
-rwxr-xr-xlibtool7621
-rwxr-xr-xltmain.sh6956
-rwxr-xr-xm4/.keep0
-rwxr-xr-xmissing367
-rwxr-xr-xprog.IAB1071
-rwxr-xr-xprog.IAD5
-rwxr-xr-xprog.IMB466
-rwxr-xr-xprog.IMD2
-rwxr-xr-xprog.PFI2
-rwxr-xr-xprog.PO1
-rwxr-xr-xprog.PR14
-rwxr-xr-xprog.PRI219
-rwxr-xr-xprog.PS979
-rwxr-xr-xprog.SearchResults3
-rwxr-xr-xprog.WK39
-rwxr-xr-xsrc/lowntfs-3g.c4492
-rwxr-xr-xsrc/ntfs-3g.8.in349
-rwxr-xr-xsrc/ntfs-3g.c4461
-rwxr-xr-xsrc/ntfs-3g.probe.8.in81
-rwxr-xr-xsrc/ntfs-3g.probe.c163
-rwxr-xr-xsrc/ntfs-3g.secaudit.8.in171
-rwxr-xr-xsrc/ntfs-3g.usermap.8.in96
-rwxr-xr-xsrc/secaudit.c7176
-rwxr-xr-xsrc/secaudit.h731
-rwxr-xr-xsrc/test.c88
-rwxr-xr-xsrc/usermap.c1356
-rwxr-xr-xstamp-h11
148 files changed, 141286 insertions, 0 deletions
diff --git a/libntfs-3g/index.c b/libntfs-3g/index.c
new file mode 100755
index 0000000..7df0dee
--- a/dev/null
+++ b/libntfs-3g/index.c
@@ -0,0 +1,2063 @@
+/**
+ * index.c - NTFS index handling. Originated from the Linux-NTFS project.
+ *
+ * Copyright (c) 2004-2005 Anton Altaparmakov
+ * Copyright (c) 2004-2005 Richard Russon
+ * Copyright (c) 2005-2006 Yura Pakhuchiy
+ * Copyright (c) 2005-2008 Szabolcs Szakacsits
+ * Copyright (c) 2007 Jean-Pierre Andre
+ *
+ * This program/include file is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as published
+ * by the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program/include file is distributed in the hope that it will be
+ * useful, but WITHOUT ANY WARRANTY; without even the implied warranty
+ * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program (in the main directory of the NTFS-3G
+ * distribution in the file COPYING); if not, write to the Free Software
+ * Foundation,Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#ifdef HAVE_STDLIB_H
+#include <stdlib.h>
+#endif
+#ifdef HAVE_STRING_H
+#include <string.h>
+#endif
+#ifdef HAVE_ERRNO_H
+#include <errno.h>
+#endif
+
+#include "attrib.h"
+#include "debug.h"
+#include "index.h"
+#include "collate.h"
+#include "mst.h"
+#include "dir.h"
+#include "logging.h"
+#include "bitmap.h"
+#include "reparse.h"
+#include "misc.h"
+
+/**
+ * ntfs_index_entry_mark_dirty - mark an index entry dirty
+ * @ictx: ntfs index context describing the index entry
+ *
+ * Mark the index entry described by the index entry context @ictx dirty.
+ *
+ * If the index entry is in the index root attribute, simply mark the inode
+ * containing the index root attribute dirty. This ensures the mftrecord, and
+ * hence the index root attribute, will be written out to disk later.
+ *
+ * If the index entry is in an index block belonging to the index allocation
+ * attribute, set ib_dirty to TRUE, thus index block will be updated during
+ * ntfs_index_ctx_put.
+ */
+void ntfs_index_entry_mark_dirty(ntfs_index_context *ictx)
+{
+ if (ictx->is_in_root)
+ ntfs_inode_mark_dirty(ictx->actx->ntfs_ino);
+ else
+ ictx->ib_dirty = TRUE;
+}
+
+static s64 ntfs_ib_vcn_to_pos(ntfs_index_context *icx, VCN vcn)
+{
+ return vcn << icx->vcn_size_bits;
+}
+
+static VCN ntfs_ib_pos_to_vcn(ntfs_index_context *icx, s64 pos)
+{
+ return pos >> icx->vcn_size_bits;
+}
+
+static int ntfs_ib_write(ntfs_index_context *icx, INDEX_BLOCK *ib)
+{
+ s64 ret, vcn = sle64_to_cpu(ib->index_block_vcn);
+
+ ntfs_log_trace("vcn: %lld\n", (long long)vcn);
+
+ ret = ntfs_attr_mst_pwrite(icx->ia_na, ntfs_ib_vcn_to_pos(icx, vcn),
+ 1, icx->block_size, ib);
+ if (ret != 1) {
+ ntfs_log_perror("Failed to write index block %lld, inode %llu",
+ (long long)vcn, (unsigned long long)icx->ni->mft_no);
+ return STATUS_ERROR;
+ }
+
+ return STATUS_OK;
+}
+
+static int ntfs_icx_ib_write(ntfs_index_context *icx)
+{
+ if (ntfs_ib_write(icx, icx->ib))
+ return STATUS_ERROR;
+
+ icx->ib_dirty = FALSE;
+
+ return STATUS_OK;
+}
+
+/**
+ * ntfs_index_ctx_get - allocate and initialize a new index context
+ * @ni: ntfs inode with which to initialize the context
+ * @name: name of the which context describes
+ * @name_len: length of the index name
+ *
+ * Allocate a new index context, initialize it with @ni and return it.
+ * Return NULL if allocation failed.
+ */
+ntfs_index_context *ntfs_index_ctx_get(ntfs_inode *ni,
+ ntfschar *name, u32 name_len)
+{
+ ntfs_index_context *icx;
+
+ ntfs_log_trace("Entering\n");
+
+ if (!ni) {
+ errno = EINVAL;
+ return NULL;
+ }
+ if (ni->nr_extents == -1)
+ ni = ni->base_ni;
+ icx = ntfs_calloc(sizeof(ntfs_index_context));
+ if (icx)
+ *icx = (ntfs_index_context) {
+ .ni = ni,
+ .name = name,
+ .name_len = name_len,
+ };
+ return icx;
+}
+
+static void ntfs_index_ctx_free(ntfs_index_context *icx)
+{
+ ntfs_log_trace("Entering\n");
+
+ if (!icx->entry)
+ return;
+
+ if (icx->actx)
+ ntfs_attr_put_search_ctx(icx->actx);
+
+ if (!icx->is_in_root) {
+ if (icx->ib_dirty) {
+ /* FIXME: Error handling!!! */
+ ntfs_ib_write(icx, icx->ib);
+ }
+ free(icx->ib);
+ }
+
+ ntfs_attr_close(icx->ia_na);
+}
+
+/**
+ * ntfs_index_ctx_put - release an index context
+ * @icx: index context to free
+ *
+ * Release the index context @icx, releasing all associated resources.
+ */
+void ntfs_index_ctx_put(ntfs_index_context *icx)
+{
+ ntfs_index_ctx_free(icx);
+ free(icx);
+}
+
+/**
+ * ntfs_index_ctx_reinit - reinitialize an index context
+ * @icx: index context to reinitialize
+ *
+ * Reinitialize the index context @icx so it can be used for ntfs_index_lookup.
+ */
+void ntfs_index_ctx_reinit(ntfs_index_context *icx)
+{
+ ntfs_log_trace("Entering\n");
+
+ ntfs_index_ctx_free(icx);
+
+ *icx = (ntfs_index_context) {
+ .ni = icx->ni,
+ .name = icx->name,
+ .name_len = icx->name_len,
+ };
+}
+
+static VCN *ntfs_ie_get_vcn_addr(INDEX_ENTRY *ie)
+{
+ return (VCN *)((u8 *)ie + le16_to_cpu(ie->length) - sizeof(VCN));
+}
+
+/**
+ * Get the subnode vcn to which the index entry refers.
+ */
+VCN ntfs_ie_get_vcn(INDEX_ENTRY *ie)
+{
+ return sle64_to_cpup(ntfs_ie_get_vcn_addr(ie));
+}
+
+static INDEX_ENTRY *ntfs_ie_get_first(INDEX_HEADER *ih)
+{
+ return (INDEX_ENTRY *)((u8 *)ih + le32_to_cpu(ih->entries_offset));
+}
+
+static INDEX_ENTRY *ntfs_ie_get_next(INDEX_ENTRY *ie)
+{
+ return (INDEX_ENTRY *)((char *)ie + le16_to_cpu(ie->length));
+}
+
+static u8 *ntfs_ie_get_end(INDEX_HEADER *ih)
+{
+ /* FIXME: check if it isn't overflowing the index block size */
+ return (u8 *)ih + le32_to_cpu(ih->index_length);
+}
+
+static int ntfs_ie_end(INDEX_ENTRY *ie)
+{
+ return ie->ie_flags & INDEX_ENTRY_END || !ie->length;
+}
+
+/**
+ * Find the last entry in the index block
+ */
+static INDEX_ENTRY *ntfs_ie_get_last(INDEX_ENTRY *ie, char *ies_end)
+{
+ ntfs_log_trace("Entering\n");
+
+ while ((char *)ie < ies_end && !ntfs_ie_end(ie))
+ ie = ntfs_ie_get_next(ie);
+
+ return ie;
+}
+
+static INDEX_ENTRY *ntfs_ie_get_by_pos(INDEX_HEADER *ih, int pos)
+{
+ INDEX_ENTRY *ie;
+
+ ntfs_log_trace("pos: %d\n", pos);
+
+ ie = ntfs_ie_get_first(ih);
+
+ while (pos-- > 0)
+ ie = ntfs_ie_get_next(ie);
+
+ return ie;
+}
+
+static INDEX_ENTRY *ntfs_ie_prev(INDEX_HEADER *ih, INDEX_ENTRY *ie)
+{
+ INDEX_ENTRY *ie_prev = NULL;
+ INDEX_ENTRY *tmp;
+
+ ntfs_log_trace("Entering\n");
+
+ tmp = ntfs_ie_get_first(ih);
+
+ while (tmp != ie) {
+ ie_prev = tmp;
+ tmp = ntfs_ie_get_next(tmp);
+ }
+
+ return ie_prev;
+}
+
+char *ntfs_ie_filename_get(INDEX_ENTRY *ie)
+{
+ FILE_NAME_ATTR *fn;
+
+ fn = (FILE_NAME_ATTR *)&ie->key;
+ return ntfs_attr_name_get(fn->file_name, fn->file_name_length);
+}
+
+void ntfs_ie_filename_dump(INDEX_ENTRY *ie)
+{
+ char *s;
+
+ s = ntfs_ie_filename_get(ie);
+ ntfs_log_debug("'%s' ", s);
+ ntfs_attr_name_free(&s);
+}
+
+void ntfs_ih_filename_dump(INDEX_HEADER *ih)
+{
+ INDEX_ENTRY *ie;
+
+ ntfs_log_trace("Entering\n");
+
+ ie = ntfs_ie_get_first(ih);
+ while (!ntfs_ie_end(ie)) {
+ ntfs_ie_filename_dump(ie);
+ ie = ntfs_ie_get_next(ie);
+ }
+}
+
+static int ntfs_ih_numof_entries(INDEX_HEADER *ih)
+{
+ int n;
+ INDEX_ENTRY *ie;
+ u8 *end;
+
+ ntfs_log_trace("Entering\n");
+
+ end = ntfs_ie_get_end(ih);
+ ie = ntfs_ie_get_first(ih);
+ for (n = 0; !ntfs_ie_end(ie) && (u8 *)ie < end; n++)
+ ie = ntfs_ie_get_next(ie);
+ return n;
+}
+
+static int ntfs_ih_one_entry(INDEX_HEADER *ih)
+{
+ return (ntfs_ih_numof_entries(ih) == 1);
+}
+
+static int ntfs_ih_zero_entry(INDEX_HEADER *ih)
+{
+ return (ntfs_ih_numof_entries(ih) == 0);
+}
+
+static void ntfs_ie_delete(INDEX_HEADER *ih, INDEX_ENTRY *ie)
+{
+ u32 new_size;
+
+ ntfs_log_trace("Entering\n");
+
+ new_size = le32_to_cpu(ih->index_length) - le16_to_cpu(ie->length);
+ ih->index_length = cpu_to_le32(new_size);
+ memmove(ie, (u8 *)ie + le16_to_cpu(ie->length),
+ new_size - ((u8 *)ie - (u8 *)ih));
+}
+
+static void ntfs_ie_set_vcn(INDEX_ENTRY *ie, VCN vcn)
+{
+ *ntfs_ie_get_vcn_addr(ie) = cpu_to_le64(vcn);
+}
+
+/**
+ * Insert @ie index entry at @pos entry. Used @ih values should be ok already.
+ */
+static void ntfs_ie_insert(INDEX_HEADER *ih, INDEX_ENTRY *ie, INDEX_ENTRY *pos)
+{
+ int ie_size = le16_to_cpu(ie->length);
+
+ ntfs_log_trace("Entering\n");
+
+ ih->index_length = cpu_to_le32(le32_to_cpu(ih->index_length) + ie_size);
+ memmove((u8 *)pos + ie_size, pos,
+ le32_to_cpu(ih->index_length) - ((u8 *)pos - (u8 *)ih) - ie_size);
+ memcpy(pos, ie, ie_size);
+}
+
+static INDEX_ENTRY *ntfs_ie_dup(INDEX_ENTRY *ie)
+{
+ INDEX_ENTRY *dup;
+
+ ntfs_log_trace("Entering\n");
+
+ dup = ntfs_malloc(le16_to_cpu(ie->length));
+ if (dup)
+ memcpy(dup, ie, le16_to_cpu(ie->length));
+
+ return dup;
+}
+
+static INDEX_ENTRY *ntfs_ie_dup_novcn(INDEX_ENTRY *ie)
+{
+ INDEX_ENTRY *dup;
+ int size = le16_to_cpu(ie->length);
+
+ ntfs_log_trace("Entering\n");
+
+ if (ie->ie_flags & INDEX_ENTRY_NODE)
+ size -= sizeof(VCN);
+
+ dup = ntfs_malloc(size);
+ if (dup) {
+ memcpy(dup, ie, size);
+ dup->ie_flags &= ~INDEX_ENTRY_NODE;
+ dup->length = cpu_to_le16(size);
+ }
+ return dup;
+}
+
+static int ntfs_ia_check(ntfs_index_context *icx, INDEX_BLOCK *ib, VCN vcn)
+{
+ u32 ib_size = (unsigned)le32_to_cpu(ib->index.allocated_size) + 0x18;
+
+ ntfs_log_trace("Entering\n");
+
+ if (!ntfs_is_indx_record(ib->magic)) {
+
+ ntfs_log_error("Corrupt index block signature: vcn %lld inode "
+ "%llu\n", (long long)vcn,
+ (unsigned long long)icx->ni->mft_no);
+ return -1;
+ }
+
+ if (sle64_to_cpu(ib->index_block_vcn) != vcn) {
+
+ ntfs_log_error("Corrupt index block: VCN (%lld) is different "
+ "from expected VCN (%lld) in inode %llu\n",
+ (long long)sle64_to_cpu(ib->index_block_vcn),
+ (long long)vcn,
+ (unsigned long long)icx->ni->mft_no);
+ return -1;
+ }
+
+ if (ib_size != icx->block_size) {
+
+ ntfs_log_error("Corrupt index block : VCN (%lld) of inode %llu "
+ "has a size (%u) differing from the index "
+ "specified size (%u)\n", (long long)vcn,
+ (unsigned long long)icx->ni->mft_no, ib_size,
+ icx->block_size);
+ return -1;
+ }
+ return 0;
+}
+
+static INDEX_ROOT *ntfs_ir_lookup(ntfs_inode *ni, ntfschar *name,
+ u32 name_len, ntfs_attr_search_ctx **ctx)
+{
+ ATTR_RECORD *a;
+ INDEX_ROOT *ir = NULL;
+
+ ntfs_log_trace("Entering\n");
+
+ *ctx = ntfs_attr_get_search_ctx(ni, NULL);
+ if (!*ctx)
+ return NULL;
+
+ if (ntfs_attr_lookup(AT_INDEX_ROOT, name, name_len, CASE_SENSITIVE,
+ 0, NULL, 0, *ctx)) {
+ ntfs_log_perror("Failed to lookup $INDEX_ROOT");
+ goto err_out;
+ }
+
+ a = (*ctx)->attr;
+ if (a->non_resident) {
+ errno = EINVAL;
+ ntfs_log_perror("Non-resident $INDEX_ROOT detected");
+ goto err_out;
+ }
+
+ ir = (INDEX_ROOT *)((char *)a + le16_to_cpu(a->value_offset));
+err_out:
+ if (!ir) {
+ ntfs_attr_put_search_ctx(*ctx);
+ *ctx = NULL;
+ }
+ return ir;
+}
+
+static INDEX_ROOT *ntfs_ir_lookup2(ntfs_inode *ni, ntfschar *name, u32 len)
+{
+ ntfs_attr_search_ctx *ctx;
+ INDEX_ROOT *ir;
+
+ ir = ntfs_ir_lookup(ni, name, len, &ctx);
+ if (ir)
+ ntfs_attr_put_search_ctx(ctx);
+ return ir;
+}
+
+/**
+ * Find a key in the index block.
+ *
+ * Return values:
+ * STATUS_OK with errno set to ESUCCESS if we know for sure that the
+ * entry exists and @ie_out points to this entry.
+ * STATUS_NOT_FOUND with errno set to ENOENT if we know for sure the
+ * entry doesn't exist and @ie_out is the insertion point.
+ * STATUS_KEEP_SEARCHING if we can't answer the above question and
+ * @vcn will contain the node index block.
+ * STATUS_ERROR with errno set if on unexpected error during lookup.
+ */
+static int ntfs_ie_lookup(const void *key, const int key_len,
+ ntfs_index_context *icx, INDEX_HEADER *ih,
+ VCN *vcn, INDEX_ENTRY **ie_out)
+{
+ INDEX_ENTRY *ie;
+ u8 *index_end;
+ int rc, item = 0;
+
+ ntfs_log_trace("Entering\n");
+
+ index_end = ntfs_ie_get_end(ih);
+
+ /*
+ * Loop until we exceed valid memory (corruption case) or until we
+ * reach the last entry.
+ */
+ for (ie = ntfs_ie_get_first(ih); ; ie = ntfs_ie_get_next(ie)) {
+ /* Bounds checks. */
+ if ((u8 *)ie + sizeof(INDEX_ENTRY_HEADER) > index_end ||
+ (u8 *)ie + le16_to_cpu(ie->length) > index_end) {
+ errno = ERANGE;
+ ntfs_log_error("Index entry out of bounds in inode "
+ "%llu.\n",
+ (unsigned long long)icx->ni->mft_no);
+ return STATUS_ERROR;
+ }
+ /*
+ * The last entry cannot contain a key. It can however contain
+ * a pointer to a child node in the B+tree so we just break out.
+ */
+ if (ntfs_ie_end(ie))
+ break;
+ /*
+ * Not a perfect match, need to do full blown collation so we
+ * know which way in the B+tree we have to go.
+ */
+ if (!icx->collate) {
+ ntfs_log_error("Collation function not defined\n");
+ errno = EOPNOTSUPP;
+ return STATUS_ERROR;
+ }
+ rc = icx->collate(icx->ni->vol, key, key_len,
+ &ie->key, le16_to_cpu(ie->key_length));
+ if (rc == NTFS_COLLATION_ERROR) {
+ ntfs_log_error("Collation error. Perhaps a filename "
+ "contains invalid characters?\n");
+ errno = ERANGE;
+ return STATUS_ERROR;
+ }
+ /*
+ * If @key collates before the key of the current entry, there
+ * is definitely no such key in this index but we might need to
+ * descend into the B+tree so we just break out of the loop.
+ */
+ if (rc == -1)
+ break;
+
+ if (!rc) {
+ *ie_out = ie;
+ errno = 0;
+ icx->parent_pos[icx->pindex] = item;
+ return STATUS_OK;
+ }
+
+ item++;
+ }
+ /*
+ * We have finished with this index block without success. Check for the
+ * presence of a child node and if not present return with errno ENOENT,
+ * otherwise we will keep searching in another index block.
+ */
+ if (!(ie->ie_flags & INDEX_ENTRY_NODE)) {
+ ntfs_log_debug("Index entry wasn't found.\n");
+ *ie_out = ie;
+ errno = ENOENT;
+ return STATUS_NOT_FOUND;
+ }
+
+ /* Get the starting vcn of the index_block holding the child node. */
+ *vcn = ntfs_ie_get_vcn(ie);
+ if (*vcn < 0) {
+ errno = EINVAL;
+ ntfs_log_perror("Negative vcn in inode %llu",
+ (unsigned long long)icx->ni->mft_no);
+ return STATUS_ERROR;
+ }
+
+ ntfs_log_trace("Parent entry number %d\n", item);
+ icx->parent_pos[icx->pindex] = item;
+
+ return STATUS_KEEP_SEARCHING;
+}
+
+static ntfs_attr *ntfs_ia_open(ntfs_index_context *icx, ntfs_inode *ni)
+{
+ ntfs_attr *na;
+
+ na = ntfs_attr_open(ni, AT_INDEX_ALLOCATION, icx->name, icx->name_len);
+ if (!na) {
+ ntfs_log_perror("Failed to open index allocation of inode "
+ "%llu", (unsigned long long)ni->mft_no);
+ return NULL;
+ }
+
+ return na;
+}
+
+static int ntfs_ib_read(ntfs_index_context *icx, VCN vcn, INDEX_BLOCK *dst)
+{
+ s64 pos, ret;
+
+ ntfs_log_trace("vcn: %lld\n", (long long)vcn);
+
+ pos = ntfs_ib_vcn_to_pos(icx, vcn);
+
+ ret = ntfs_attr_mst_pread(icx->ia_na, pos, 1, icx->block_size, (u8 *)dst);
+ if (ret != 1) {
+ if (ret == -1)
+ ntfs_log_perror("Failed to read index block");
+ else
+ ntfs_log_error("Failed to read full index block at "
+ "%lld\n", (long long)pos);
+ return -1;
+ }
+
+ if (ntfs_ia_check(icx, dst, vcn))
+ return -1;
+
+ return 0;
+}
+
+static int ntfs_icx_parent_inc(ntfs_index_context *icx)
+{
+ icx->pindex++;
+ if (icx->pindex >= MAX_PARENT_VCN) {
+ errno = EOPNOTSUPP;
+ ntfs_log_perror("Index is over %d level deep", MAX_PARENT_VCN);
+ return STATUS_ERROR;
+ }
+ return STATUS_OK;
+}
+
+static int ntfs_icx_parent_dec(ntfs_index_context *icx)
+{
+ icx->pindex--;
+ if (icx->pindex < 0) {
+ errno = EINVAL;
+ ntfs_log_perror("Corrupt index pointer (%d)", icx->pindex);
+ return STATUS_ERROR;
+ }
+ return STATUS_OK;
+}
+
+/**
+ * ntfs_index_lookup - find a key in an index and return its index entry
+ * @key: [IN] key for which to search in the index
+ * @key_len: [IN] length of @key in bytes
+ * @icx: [IN/OUT] context describing the index and the returned entry
+ *
+ * Before calling ntfs_index_lookup(), @icx must have been obtained from a
+ * call to ntfs_index_ctx_get().
+ *
+ * Look for the @key in the index specified by the index lookup context @icx.
+ * ntfs_index_lookup() walks the contents of the index looking for the @key.
+ *
+ * If the @key is found in the index, 0 is returned and @icx is setup to
+ * describe the index entry containing the matching @key. @icx->entry is the
+ * index entry and @icx->data and @icx->data_len are the index entry data and
+ * its length in bytes, respectively.
+ *
+ * If the @key is not found in the index, -1 is returned, errno = ENOENT and
+ * @icx is setup to describe the index entry whose key collates immediately
+ * after the search @key, i.e. this is the position in the index at which
+ * an index entry with a key of @key would need to be inserted.
+ *
+ * If an error occurs return -1, set errno to error code and @icx is left
+ * untouched.
+ *
+ * When finished with the entry and its data, call ntfs_index_ctx_put() to free
+ * the context and other associated resources.
+ *
+ * If the index entry was modified, call ntfs_index_entry_mark_dirty() before
+ * the call to ntfs_index_ctx_put() to ensure that the changes are written
+ * to disk.
+ */
+int ntfs_index_lookup(const void *key, const int key_len, ntfs_index_context *icx)
+{
+ VCN old_vcn, vcn;
+ ntfs_inode *ni = icx->ni;
+ INDEX_ROOT *ir;
+ INDEX_ENTRY *ie;
+ INDEX_BLOCK *ib = NULL;
+ int ret, err = 0;
+
+ ntfs_log_trace("Entering\n");
+
+ if (!key || key_len <= 0) {
+ errno = EINVAL;
+ ntfs_log_perror("key: %p key_len: %d", key, key_len);
+ return -1;
+ }
+
+ ir = ntfs_ir_lookup(ni, icx->name, icx->name_len, &icx->actx);
+ if (!ir) {
+ if (errno == ENOENT)
+ errno = EIO;
+ return -1;
+ }
+
+ icx->block_size = le32_to_cpu(ir->index_block_size);
+ if (icx->block_size < NTFS_BLOCK_SIZE) {
+ errno = EINVAL;
+ ntfs_log_perror("Index block size (%d) is smaller than the "
+ "sector size (%d)", icx->block_size, NTFS_BLOCK_SIZE);
+ goto err_out;
+ }
+
+ if (ni->vol->cluster_size <= icx->block_size)
+ icx->vcn_size_bits = ni->vol->cluster_size_bits;
+ else
+ icx->vcn_size_bits = ni->vol->sector_size_bits;
+ /* get the appropriate collation function */
+ icx->collate = ntfs_get_collate_function(ir->collation_rule);
+ if (!icx->collate) {
+ err = errno = EOPNOTSUPP;
+ ntfs_log_perror("Unknown collation rule 0x%x",
+ (unsigned)le32_to_cpu(ir->collation_rule));
+ goto err_out;
+ }
+
+ old_vcn = VCN_INDEX_ROOT_PARENT;
+ /*
+ * FIXME: check for both ir and ib that the first index entry is
+ * within the index block.
+ */
+ ret = ntfs_ie_lookup(key, key_len, icx, &ir->index, &vcn, &ie);
+ if (ret == STATUS_ERROR) {
+ err = errno;
+ goto err_out;
+ }
+
+ icx->ir = ir;
+
+ if (ret != STATUS_KEEP_SEARCHING) {
+ /* STATUS_OK or STATUS_NOT_FOUND */
+ err = errno;
+ icx->is_in_root = TRUE;
+ icx->parent_vcn[icx->pindex] = old_vcn;
+ goto done;
+ }
+
+ /* Child node present, descend into it. */
+
+ icx->ia_na = ntfs_ia_open(icx, ni);
+ if (!icx->ia_na)
+ goto err_out;
+
+ ib = ntfs_malloc(icx->block_size);
+ if (!ib) {
+ err = errno;
+ goto err_out;
+ }
+
+descend_into_child_node:
+
+ icx->parent_vcn[icx->pindex] = old_vcn;
+ if (ntfs_icx_parent_inc(icx)) {
+ err = errno;
+ goto err_out;
+ }
+ old_vcn = vcn;
+
+ ntfs_log_debug("Descend into node with VCN %lld\n", (long long)vcn);
+
+ if (ntfs_ib_read(icx, vcn, ib))
+ goto err_out;
+
+ ret = ntfs_ie_lookup(key, key_len, icx, &ib->index, &vcn, &ie);
+ if (ret != STATUS_KEEP_SEARCHING) {
+ err = errno;
+ if (ret == STATUS_ERROR)
+ goto err_out;
+
+ /* STATUS_OK or STATUS_NOT_FOUND */
+ icx->is_in_root = FALSE;
+ icx->ib = ib;
+ icx->parent_vcn[icx->pindex] = vcn;
+ goto done;
+ }
+
+ if ((ib->index.ih_flags & NODE_MASK) == LEAF_NODE) {
+ ntfs_log_error("Index entry with child node found in a leaf "
+ "node in inode 0x%llx.\n",
+ (unsigned long long)ni->mft_no);
+ goto err_out;
+ }
+
+ goto descend_into_child_node;
+err_out:
+ free(ib);
+ if (!err)
+ err = EIO;
+ errno = err;
+ return -1;
+done:
+ icx->entry = ie;
+ icx->data = (u8 *)ie + offsetof(INDEX_ENTRY, key);
+ icx->data_len = le16_to_cpu(ie->key_length);
+ ntfs_log_trace("Done.\n");
+ if (err) {
+ errno = err;
+ return -1;
+ }
+ return 0;
+
+}
+
+static INDEX_BLOCK *ntfs_ib_alloc(VCN ib_vcn, u32 ib_size,
+ INDEX_HEADER_FLAGS node_type)
+{
+ INDEX_BLOCK *ib;
+ int ih_size = sizeof(INDEX_HEADER);
+
+ ntfs_log_trace("ib_vcn: %lld ib_size: %u\n", (long long)ib_vcn, ib_size);
+
+ ib = ntfs_calloc(ib_size);
+ if (!ib)
+ return NULL;
+
+ ib->magic = magic_INDX;
+ ib->usa_ofs = cpu_to_le16(sizeof(INDEX_BLOCK));
+ ib->usa_count = cpu_to_le16(ib_size / NTFS_BLOCK_SIZE + 1);
+ /* Set USN to 1 */
+ *(u16 *)((char *)ib + le16_to_cpu(ib->usa_ofs)) = cpu_to_le16(1);
+ ib->lsn = cpu_to_le64(0);
+
+ ib->index_block_vcn = cpu_to_sle64(ib_vcn);
+
+ ib->index.entries_offset = cpu_to_le32((ih_size +
+ le16_to_cpu(ib->usa_count) * 2 + 7) & ~7);
+ ib->index.index_length = 0;
+ ib->index.allocated_size = cpu_to_le32(ib_size -
+ (sizeof(INDEX_BLOCK) - ih_size));
+ ib->index.ih_flags = node_type;
+
+ return ib;
+}
+
+/**
+ * Find the median by going through all the entries
+ */
+static INDEX_ENTRY *ntfs_ie_get_median(INDEX_HEADER *ih)
+{
+ INDEX_ENTRY *ie, *ie_start;
+ u8 *ie_end;
+ int i = 0, median;
+
+ ntfs_log_trace("Entering\n");
+
+ ie = ie_start = ntfs_ie_get_first(ih);
+ ie_end = (u8 *)ntfs_ie_get_end(ih);
+
+ while ((u8 *)ie < ie_end && !ntfs_ie_end(ie)) {
+ ie = ntfs_ie_get_next(ie);
+ i++;
+ }
+ /*
+ * NOTE: this could be also the entry at the half of the index block.
+ */
+ median = i / 2 - 1;
+
+ ntfs_log_trace("Entries: %d median: %d\n", i, median);
+
+ for (i = 0, ie = ie_start; i <= median; i++)
+ ie = ntfs_ie_get_next(ie);
+
+ return ie;
+}
+
+static s64 ntfs_ibm_vcn_to_pos(ntfs_index_context *icx, VCN vcn)
+{
+ return ntfs_ib_vcn_to_pos(icx, vcn) / icx->block_size;
+}
+
+static s64 ntfs_ibm_pos_to_vcn(ntfs_index_context *icx, s64 pos)
+{
+ return ntfs_ib_pos_to_vcn(icx, pos * icx->block_size);
+}
+
+static int ntfs_ibm_add(ntfs_index_context *icx)
+{
+ u8 bmp[8];
+
+ ntfs_log_trace("Entering\n");
+
+ if (ntfs_attr_exist(icx->ni, AT_BITMAP, icx->name, icx->name_len))
+ return STATUS_OK;
+ /*
+ * AT_BITMAP must be at least 8 bytes.
+ */
+ memset(bmp, 0, sizeof(bmp));
+ if (ntfs_attr_add(icx->ni, AT_BITMAP, icx->name, icx->name_len,
+ bmp, sizeof(bmp))) {
+ ntfs_log_perror("Failed to add AT_BITMAP");
+ return STATUS_ERROR;
+ }
+
+ return STATUS_OK;
+}
+
+static int ntfs_ibm_modify(ntfs_index_context *icx, VCN vcn, int set)
+{
+ u8 byte;
+ s64 pos = ntfs_ibm_vcn_to_pos(icx, vcn);
+ u32 bpos = pos / 8;
+ u32 bit = 1 << (pos % 8);
+ ntfs_attr *na;
+ int ret = STATUS_ERROR;
+
+ ntfs_log_trace("%s vcn: %lld\n", set ? "set" : "clear", (long long)vcn);
+
+ na = ntfs_attr_open(icx->ni, AT_BITMAP, icx->name, icx->name_len);
+ if (!na) {
+ ntfs_log_perror("Failed to open $BITMAP attribute");
+ return -1;
+ }
+
+ if (set) {
+ if (na->data_size < bpos + 1) {
+ if (ntfs_attr_truncate(na, (na->data_size + 8) & ~7)) {
+ ntfs_log_perror("Failed to truncate AT_BITMAP");
+ goto err_na;
+ }
+ }
+ }
+
+ if (ntfs_attr_pread(na, bpos, 1, &byte) != 1) {
+ ntfs_log_perror("Failed to read $BITMAP");
+ goto err_na;
+ }
+
+ if (set)
+ byte |= bit;
+ else
+ byte &= ~bit;
+
+ if (ntfs_attr_pwrite(na, bpos, 1, &byte) != 1) {
+ ntfs_log_perror("Failed to write $Bitmap");
+ goto err_na;
+ }
+
+ ret = STATUS_OK;
+err_na:
+ ntfs_attr_close(na);
+ return ret;
+}
+
+
+static int ntfs_ibm_set(ntfs_index_context *icx, VCN vcn)
+{
+ return ntfs_ibm_modify(icx, vcn, 1);
+}
+
+static int ntfs_ibm_clear(ntfs_index_context *icx, VCN vcn)
+{
+ return ntfs_ibm_modify(icx, vcn, 0);
+}
+
+static VCN ntfs_ibm_get_free(ntfs_index_context *icx)
+{
+ u8 *bm;
+ int bit;
+ s64 vcn, byte, size;
+
+ ntfs_log_trace("Entering\n");
+
+ bm = ntfs_attr_readall(icx->ni, AT_BITMAP, icx->name, icx->name_len,
+ &size);
+ if (!bm)
+ return (VCN)-1;
+
+ for (byte = 0; byte < size; byte++) {
+
+ if (bm[byte] == 255)
+ continue;
+
+ for (bit = 0; bit < 8; bit++) {
+ if (!(bm[byte] & (1 << bit))) {
+ vcn = ntfs_ibm_pos_to_vcn(icx, byte * 8 + bit);
+ goto out;
+ }
+ }
+ }
+
+ vcn = ntfs_ibm_pos_to_vcn(icx, size * 8);
+out:
+ ntfs_log_trace("allocated vcn: %lld\n", (long long)vcn);
+
+ if (ntfs_ibm_set(icx, vcn))
+ vcn = (VCN)-1;
+
+ free(bm);
+ return vcn;
+}
+
+static INDEX_BLOCK *ntfs_ir_to_ib(INDEX_ROOT *ir, VCN ib_vcn)
+{
+ INDEX_BLOCK *ib;
+ INDEX_ENTRY *ie_last;
+ char *ies_start, *ies_end;
+ int i;
+
+ ntfs_log_trace("Entering\n");
+
+ ib = ntfs_ib_alloc(ib_vcn, le32_to_cpu(ir->index_block_size), LEAF_NODE);
+ if (!ib)
+ return NULL;
+
+ ies_start = (char *)ntfs_ie_get_first(&ir->index);
+ ies_end = (char *)ntfs_ie_get_end(&ir->index);
+ ie_last = ntfs_ie_get_last((INDEX_ENTRY *)ies_start, ies_end);
+ /*
+ * Copy all entries, including the termination entry
+ * as well, which can never have any data.
+ */
+ i = (char *)ie_last - ies_start + le16_to_cpu(ie_last->length);
+ memcpy(ntfs_ie_get_first(&ib->index), ies_start, i);
+
+ ib->index.ih_flags = ir->index.ih_flags;
+ ib->index.index_length = cpu_to_le32(i +
+ le32_to_cpu(ib->index.entries_offset));
+ return ib;
+}
+
+static void ntfs_ir_nill(INDEX_ROOT *ir)
+{
+ INDEX_ENTRY *ie_last;
+ char *ies_start, *ies_end;
+
+ ntfs_log_trace("Entering\n");
+ /*
+ * TODO: This function could be much simpler.
+ */
+ ies_start = (char *)ntfs_ie_get_first(&ir->index);
+ ies_end = (char *)ntfs_ie_get_end(&ir->index);
+ ie_last = ntfs_ie_get_last((INDEX_ENTRY *)ies_start, ies_end);
+ /*
+ * Move the index root termination entry forward
+ */
+ if ((char *)ie_last > ies_start) {
+ memmove(ies_start, (char *)ie_last, le16_to_cpu(ie_last->length));
+ ie_last = (INDEX_ENTRY *)ies_start;
+ }
+}
+
+static int ntfs_ib_copy_tail(ntfs_index_context *icx, INDEX_BLOCK *src,
+ INDEX_ENTRY *median, VCN new_vcn)
+{
+ u8 *ies_end;
+ INDEX_ENTRY *ie_head; /* first entry after the median */
+ int tail_size, ret;
+ INDEX_BLOCK *dst;
+
+ ntfs_log_trace("Entering\n");
+
+ dst = ntfs_ib_alloc(new_vcn, icx->block_size,
+ src->index.ih_flags & NODE_MASK);
+ if (!dst)
+ return STATUS_ERROR;
+
+ ie_head = ntfs_ie_get_next(median);
+
+ ies_end = (u8 *)ntfs_ie_get_end(&src->index);
+ tail_size = ies_end - (u8 *)ie_head;
+ memcpy(ntfs_ie_get_first(&dst->index), ie_head, tail_size);
+
+ dst->index.index_length = cpu_to_le32(tail_size +
+ le32_to_cpu(dst->index.entries_offset));
+ ret = ntfs_ib_write(icx, dst);
+
+ free(dst);
+ return ret;
+}
+
+static int ntfs_ib_cut_tail(ntfs_index_context *icx, INDEX_BLOCK *ib,
+ INDEX_ENTRY *ie)
+{
+ char *ies_start, *ies_end;
+ INDEX_ENTRY *ie_last;
+
+ ntfs_log_trace("Entering\n");
+
+ ies_start = (char *)ntfs_ie_get_first(&ib->index);
+ ies_end = (char *)ntfs_ie_get_end(&ib->index);
+
+ ie_last = ntfs_ie_get_last((INDEX_ENTRY *)ies_start, ies_end);
+ if (ie_last->ie_flags & INDEX_ENTRY_NODE)
+ ntfs_ie_set_vcn(ie_last, ntfs_ie_get_vcn(ie));
+
+ memcpy(ie, ie_last, le16_to_cpu(ie_last->length));
+
+ ib->index.index_length = cpu_to_le32(((char *)ie - ies_start) +
+ le16_to_cpu(ie->length) + le32_to_cpu(ib->index.entries_offset));
+
+ if (ntfs_ib_write(icx, ib))
+ return STATUS_ERROR;
+
+ return STATUS_OK;
+}
+
+static int ntfs_ia_add(ntfs_index_context *icx)
+{
+ ntfs_log_trace("Entering\n");
+
+ if (ntfs_ibm_add(icx))
+ return -1;
+
+ if (!ntfs_attr_exist(icx->ni, AT_INDEX_ALLOCATION, icx->name, icx->name_len)) {
+
+ if (ntfs_attr_add(icx->ni, AT_INDEX_ALLOCATION, icx->name,
+ icx->name_len, NULL, 0)) {
+ ntfs_log_perror("Failed to add AT_INDEX_ALLOCATION");
+ return -1;
+ }
+ }
+
+ icx->ia_na = ntfs_ia_open(icx, icx->ni);
+ if (!icx->ia_na)
+ return -1;
+
+ return 0;
+}
+
+static int ntfs_ir_reparent(ntfs_index_context *icx)
+{
+ ntfs_attr_search_ctx *ctx = NULL;
+ INDEX_ROOT *ir;
+ INDEX_ENTRY *ie;
+ INDEX_BLOCK *ib = NULL;
+ VCN new_ib_vcn;
+ int ret = STATUS_ERROR;
+
+ ntfs_log_trace("Entering\n");
+
+ ir = ntfs_ir_lookup2(icx->ni, icx->name, icx->name_len);
+ if (!ir)
+ goto out;
+
+ if ((ir->index.ih_flags & NODE_MASK) == SMALL_INDEX)
+ if (ntfs_ia_add(icx))
+ goto out;
+
+ new_ib_vcn = ntfs_ibm_get_free(icx);
+ if (new_ib_vcn == -1)
+ goto out;
+
+ ir = ntfs_ir_lookup2(icx->ni, icx->name, icx->name_len);
+ if (!ir)
+ goto clear_bmp;
+
+ ib = ntfs_ir_to_ib(ir, new_ib_vcn);
+ if (ib == NULL) {
+ ntfs_log_perror("Failed to move index root to index block");
+ goto clear_bmp;
+ }
+
+ if (ntfs_ib_write(icx, ib))
+ goto clear_bmp;
+
+ ir = ntfs_ir_lookup(icx->ni, icx->name, icx->name_len, &ctx);
+ if (!ir)
+ goto clear_bmp;
+
+ ntfs_ir_nill(ir);
+
+ ie = ntfs_ie_get_first(&ir->index);
+ ie->ie_flags |= INDEX_ENTRY_NODE;
+ ie->length = cpu_to_le16(sizeof(INDEX_ENTRY_HEADER) + sizeof(VCN));
+
+ ir->index.ih_flags = LARGE_INDEX;
+ ir->index.index_length = cpu_to_le32(le32_to_cpu(ir->index.entries_offset)
+ + le16_to_cpu(ie->length));
+ ir->index.allocated_size = ir->index.index_length;
+
+ if (ntfs_resident_attr_value_resize(ctx->mrec, ctx->attr,
+ sizeof(INDEX_ROOT) - sizeof(INDEX_HEADER) +
+ le32_to_cpu(ir->index.allocated_size)))
+ /* FIXME: revert index root */
+ goto clear_bmp;
+ /*
+ * FIXME: do it earlier if we have enough space in IR (should always),
+ * so in error case we wouldn't lose the IB.
+ */
+ ntfs_ie_set_vcn(ie, new_ib_vcn);
+
+ ret = STATUS_OK;
+err_out:
+ free(ib);
+ ntfs_attr_put_search_ctx(ctx);
+out:
+ return ret;
+clear_bmp:
+ ntfs_ibm_clear(icx, new_ib_vcn);
+ goto err_out;
+}
+
+/**
+ * ntfs_ir_truncate - Truncate index root attribute
+ *
+ * Returns STATUS_OK, STATUS_RESIDENT_ATTRIBUTE_FILLED_MFT or STATUS_ERROR.
+ */
+static int ntfs_ir_truncate(ntfs_index_context *icx, int data_size)
+{
+ ntfs_attr *na;
+ int ret;
+
+ ntfs_log_trace("Entering\n");
+
+ na = ntfs_attr_open(icx->ni, AT_INDEX_ROOT, icx->name, icx->name_len);
+ if (!na) {
+ ntfs_log_perror("Failed to open INDEX_ROOT");
+ return STATUS_ERROR;
+ }
+ /*
+ * INDEX_ROOT must be resident and its entries can be moved to
+ * INDEX_BLOCK, so ENOSPC isn't a real error.
+ */
+ ret = ntfs_attr_truncate(na, data_size + offsetof(INDEX_ROOT, index));
+ if (ret == STATUS_OK) {
+
+ icx->ir = ntfs_ir_lookup2(icx->ni, icx->name, icx->name_len);
+ if (!icx->ir)
+ return STATUS_ERROR;
+
+ icx->ir->index.allocated_size = cpu_to_le32(data_size);
+
+ } else if (ret == STATUS_ERROR)
+ ntfs_log_perror("Failed to truncate INDEX_ROOT");
+
+ ntfs_attr_close(na);
+ return ret;
+}
+
+/**
+ * ntfs_ir_make_space - Make more space for the index root attribute
+ *
+ * On success return STATUS_OK or STATUS_KEEP_SEARCHING.
+ * On error return STATUS_ERROR.
+ */
+static int ntfs_ir_make_space(ntfs_index_context *icx, int data_size)
+{
+ int ret;
+
+ ntfs_log_trace("Entering\n");
+
+ ret = ntfs_ir_truncate(icx, data_size);
+ if (ret == STATUS_RESIDENT_ATTRIBUTE_FILLED_MFT) {
+
+ ret = ntfs_ir_reparent(icx);
+ if (ret == STATUS_OK)
+ ret = STATUS_KEEP_SEARCHING;
+ else
+ ntfs_log_perror("Failed to nodify INDEX_ROOT");
+ }
+
+ return ret;
+}
+
+/*
+ * NOTE: 'ie' must be a copy of a real index entry.
+ */
+static int ntfs_ie_add_vcn(INDEX_ENTRY **ie)
+{
+ INDEX_ENTRY *p, *old = *ie;
+
+ old->length = cpu_to_le16(le16_to_cpu(old->length) + sizeof(VCN));
+ p = realloc(old, le16_to_cpu(old->length));
+ if (!p)
+ return STATUS_ERROR;
+
+ p->ie_flags |= INDEX_ENTRY_NODE;
+ *ie = p;
+
+ return STATUS_OK;
+}
+
+static int ntfs_ih_insert(INDEX_HEADER *ih, INDEX_ENTRY *orig_ie, VCN new_vcn,
+ int pos)
+{
+ INDEX_ENTRY *ie_node, *ie;
+ int ret = STATUS_ERROR;
+ VCN old_vcn;
+
+ ntfs_log_trace("Entering\n");
+
+ ie = ntfs_ie_dup(orig_ie);
+ if (!ie)
+ return STATUS_ERROR;
+
+ if (!(ie->ie_flags & INDEX_ENTRY_NODE))
+ if (ntfs_ie_add_vcn(&ie))
+ goto out;
+
+ ie_node = ntfs_ie_get_by_pos(ih, pos);
+ old_vcn = ntfs_ie_get_vcn(ie_node);
+ ntfs_ie_set_vcn(ie_node, new_vcn);
+
+ ntfs_ie_insert(ih, ie, ie_node);
+ ntfs_ie_set_vcn(ie_node, old_vcn);
+ ret = STATUS_OK;
+out:
+ free(ie);
+
+ return ret;
+}
+
+static VCN ntfs_icx_parent_vcn(ntfs_index_context *icx)
+{
+ return icx->parent_vcn[icx->pindex];
+}
+
+static VCN ntfs_icx_parent_pos(ntfs_index_context *icx)
+{
+ return icx->parent_pos[icx->pindex];
+}
+
+
+static int ntfs_ir_insert_median(ntfs_index_context *icx, INDEX_ENTRY *median,
+ VCN new_vcn)
+{
+ u32 new_size;
+ int ret;
+
+ ntfs_log_trace("Entering\n");
+
+ icx->ir = ntfs_ir_lookup2(icx->ni, icx->name, icx->name_len);
+ if (!icx->ir)
+ return STATUS_ERROR;
+
+ new_size = le32_to_cpu(icx->ir->index.index_length) +
+ le16_to_cpu(median->length);
+ if (!(median->ie_flags & INDEX_ENTRY_NODE))
+ new_size += sizeof(VCN);
+
+ ret = ntfs_ir_make_space(icx, new_size);
+ if (ret != STATUS_OK)
+ return ret;
+
+ icx->ir = ntfs_ir_lookup2(icx->ni, icx->name, icx->name_len);
+ if (!icx->ir)
+ return STATUS_ERROR;
+
+ return ntfs_ih_insert(&icx->ir->index, median, new_vcn,
+ ntfs_icx_parent_pos(icx));
+}
+
+static int ntfs_ib_split(ntfs_index_context *icx, INDEX_BLOCK *ib);
+
+/**
+ * On success return STATUS_OK or STATUS_KEEP_SEARCHING.
+ * On error return STATUS_ERROR.
+ */
+static int ntfs_ib_insert(ntfs_index_context *icx, INDEX_ENTRY *ie, VCN new_vcn)
+{
+ INDEX_BLOCK *ib;
+ u32 idx_size, allocated_size;
+ int err = STATUS_ERROR;
+ VCN old_vcn;
+
+ ntfs_log_trace("Entering\n");
+
+ ib = ntfs_malloc(icx->block_size);
+ if (!ib)
+ return -1;
+
+ old_vcn = ntfs_icx_parent_vcn(icx);
+
+ if (ntfs_ib_read(icx, old_vcn, ib))
+ goto err_out;
+
+ idx_size = le32_to_cpu(ib->index.index_length);
+ allocated_size = le32_to_cpu(ib->index.allocated_size);
+ /* FIXME: sizeof(VCN) should be included only if ie has no VCN */
+ if (idx_size + le16_to_cpu(ie->length) + sizeof(VCN) > allocated_size) {
+ err = ntfs_ib_split(icx, ib);
+ if (err == STATUS_OK)
+ err = STATUS_KEEP_SEARCHING;
+ goto err_out;
+ }
+
+ if (ntfs_ih_insert(&ib->index, ie, new_vcn, ntfs_icx_parent_pos(icx)))
+ goto err_out;
+
+ if (ntfs_ib_write(icx, ib))
+ goto err_out;
+
+ err = STATUS_OK;
+err_out:
+ free(ib);
+ return err;
+}
+
+/**
+ * ntfs_ib_split - Split an index block
+ *
+ * On success return STATUS_OK or STATUS_KEEP_SEARCHING.
+ * On error return is STATUS_ERROR.
+ */
+static int ntfs_ib_split(ntfs_index_context *icx, INDEX_BLOCK *ib)
+{
+ INDEX_ENTRY *median;
+ VCN new_vcn;
+ int ret;
+
+ ntfs_log_trace("Entering\n");
+
+ if (ntfs_icx_parent_dec(icx))
+ return STATUS_ERROR;
+
+ median = ntfs_ie_get_median(&ib->index);
+ new_vcn = ntfs_ibm_get_free(icx);
+ if (new_vcn == -1)
+ return STATUS_ERROR;
+
+ if (ntfs_ib_copy_tail(icx, ib, median, new_vcn)) {
+ ntfs_ibm_clear(icx, new_vcn);
+ return STATUS_ERROR;
+ }
+
+ if (ntfs_icx_parent_vcn(icx) == VCN_INDEX_ROOT_PARENT)
+ ret = ntfs_ir_insert_median(icx, median, new_vcn);
+ else
+ ret = ntfs_ib_insert(icx, median, new_vcn);
+
+ if (ret != STATUS_OK) {
+ ntfs_ibm_clear(icx, new_vcn);
+ return ret;
+ }
+
+ ret = ntfs_ib_cut_tail(icx, ib, median);
+
+ return ret;
+}
+
+/* JPA static */
+int ntfs_ie_add(ntfs_index_context *icx, INDEX_ENTRY *ie)
+{
+ INDEX_HEADER *ih;
+ int allocated_size, new_size;
+ int ret = STATUS_ERROR;
+
+#ifdef DEBUG
+/* removed by JPA to make function usable for security indexes
+ char *fn;
+ fn = ntfs_ie_filename_get(ie);
+ ntfs_log_trace("file: '%s'\n", fn);
+ ntfs_attr_name_free(&fn);
+*/
+#endif
+
+ while (1) {
+
+ if (!ntfs_index_lookup(&ie->key, le16_to_cpu(ie->key_length), icx)) {
+ errno = EEXIST;
+ ntfs_log_perror("Index already have such entry");
+ goto err_out;
+ }
+ if (errno != ENOENT) {
+ ntfs_log_perror("Failed to find place for new entry");
+ goto err_out;
+ }
+
+ if (icx->is_in_root)
+ ih = &icx->ir->index;
+ else
+ ih = &icx->ib->index;
+
+ allocated_size = le32_to_cpu(ih->allocated_size);
+ new_size = le32_to_cpu(ih->index_length) + le16_to_cpu(ie->length);
+
+ if (new_size <= allocated_size)
+ break;
+
+ ntfs_log_trace("index block sizes: allocated: %d needed: %d\n",
+ allocated_size, new_size);
+
+ if (icx->is_in_root) {
+ if (ntfs_ir_make_space(icx, new_size) == STATUS_ERROR)
+ goto err_out;
+ } else {
+ if (ntfs_ib_split(icx, icx->ib) == STATUS_ERROR)
+ goto err_out;
+ }
+
+ ntfs_inode_mark_dirty(icx->actx->ntfs_ino);
+ ntfs_index_ctx_reinit(icx);
+ }
+
+ ntfs_ie_insert(ih, ie, icx->entry);
+ ntfs_index_entry_mark_dirty(icx);
+
+ ret = STATUS_OK;
+err_out:
+ ntfs_log_trace("%s\n", ret ? "Failed" : "Done");
+ return ret;
+}
+
+/**
+ * ntfs_index_add_filename - add filename to directory index
+ * @ni: ntfs inode describing directory to which index add filename
+ * @fn: FILE_NAME attribute to add
+ * @mref: reference of the inode which @fn describes
+ *
+ * Return 0 on success or -1 on error with errno set to the error code.
+ */
+int ntfs_index_add_filename(ntfs_inode *ni, FILE_NAME_ATTR *fn, MFT_REF mref)
+{
+ INDEX_ENTRY *ie;
+ ntfs_index_context *icx;
+ int fn_size, ie_size, err, ret = -1;
+
+ ntfs_log_trace("Entering\n");
+
+ if (!ni || !fn) {
+ ntfs_log_error("Invalid arguments.\n");
+ errno = EINVAL;
+ return -1;
+ }
+
+ fn_size = (fn->file_name_length * sizeof(ntfschar)) +
+ sizeof(FILE_NAME_ATTR);
+ ie_size = (sizeof(INDEX_ENTRY_HEADER) + fn_size + 7) & ~7;
+
+ ie = ntfs_calloc(ie_size);
+ if (!ie)
+ return -1;
+
+ ie->indexed_file = cpu_to_le64(mref);
+ ie->length = cpu_to_le16(ie_size);
+ ie->key_length = cpu_to_le16(fn_size);
+ memcpy(&ie->key, fn, fn_size);
+
+ icx = ntfs_index_ctx_get(ni, NTFS_INDEX_I30, 4);
+ if (!icx)
+ goto out;
+
+ ret = ntfs_ie_add(icx, ie);
+ err = errno;
+ ntfs_index_ctx_put(icx);
+ errno = err;
+out:
+ free(ie);
+ return ret;
+}
+
+static int ntfs_ih_takeout(ntfs_index_context *icx, INDEX_HEADER *ih,
+ INDEX_ENTRY *ie, INDEX_BLOCK *ib)
+{
+ INDEX_ENTRY *ie_roam;
+ int ret = STATUS_ERROR;
+
+ ntfs_log_trace("Entering\n");
+
+ ie_roam = ntfs_ie_dup_novcn(ie);
+ if (!ie_roam)
+ return STATUS_ERROR;
+
+ ntfs_ie_delete(ih, ie);
+
+ if (ntfs_icx_parent_vcn(icx) == VCN_INDEX_ROOT_PARENT)
+ ntfs_inode_mark_dirty(icx->actx->ntfs_ino);
+ else
+ if (ntfs_ib_write(icx, ib))
+ goto out;
+
+ ntfs_index_ctx_reinit(icx);
+
+ ret = ntfs_ie_add(icx, ie_roam);
+out:
+ free(ie_roam);
+ return ret;
+}
+
+/**
+ * Used if an empty index block to be deleted has END entry as the parent
+ * in the INDEX_ROOT which is the only one there.
+ */
+static void ntfs_ir_leafify(ntfs_index_context *icx, INDEX_HEADER *ih)
+{
+ INDEX_ENTRY *ie;
+
+ ntfs_log_trace("Entering\n");
+
+ ie = ntfs_ie_get_first(ih);
+ ie->ie_flags &= ~INDEX_ENTRY_NODE;
+ ie->length = cpu_to_le16(le16_to_cpu(ie->length) - sizeof(VCN));
+
+ ih->index_length = cpu_to_le32(le32_to_cpu(ih->index_length) - sizeof(VCN));
+ ih->ih_flags &= ~LARGE_INDEX;
+
+ /* Not fatal error */
+ ntfs_ir_truncate(icx, le32_to_cpu(ih->index_length));
+}
+
+/**
+ * Used if an empty index block to be deleted has END entry as the parent
+ * in the INDEX_ROOT which is not the only one there.
+ */
+static int ntfs_ih_reparent_end(ntfs_index_context *icx, INDEX_HEADER *ih,
+ INDEX_BLOCK *ib)
+{
+ INDEX_ENTRY *ie, *ie_prev;
+
+ ntfs_log_trace("Entering\n");
+
+ ie = ntfs_ie_get_by_pos(ih, ntfs_icx_parent_pos(icx));
+ ie_prev = ntfs_ie_prev(ih, ie);
+
+ ntfs_ie_set_vcn(ie, ntfs_ie_get_vcn(ie_prev));
+
+ return ntfs_ih_takeout(icx, ih, ie_prev, ib);
+}
+
+static int ntfs_index_rm_leaf(ntfs_index_context *icx)
+{
+ INDEX_BLOCK *ib = NULL;
+ INDEX_HEADER *parent_ih;
+ INDEX_ENTRY *ie;
+ int ret = STATUS_ERROR;
+
+ ntfs_log_trace("pindex: %d\n", icx->pindex);
+
+ if (ntfs_icx_parent_dec(icx))
+ return STATUS_ERROR;
+
+ if (ntfs_ibm_clear(icx, icx->parent_vcn[icx->pindex + 1]))
+ return STATUS_ERROR;
+
+ if (ntfs_icx_parent_vcn(icx) == VCN_INDEX_ROOT_PARENT)
+ parent_ih = &icx->ir->index;
+ else {
+ ib = ntfs_malloc(icx->block_size);
+ if (!ib)
+ return STATUS_ERROR;
+
+ if (ntfs_ib_read(icx, ntfs_icx_parent_vcn(icx), ib))
+ goto out;
+
+ parent_ih = &ib->index;
+ }
+
+ ie = ntfs_ie_get_by_pos(parent_ih, ntfs_icx_parent_pos(icx));
+ if (!ntfs_ie_end(ie)) {
+ ret = ntfs_ih_takeout(icx, parent_ih, ie, ib);
+ goto out;
+ }
+
+ if (ntfs_ih_zero_entry(parent_ih)) {
+
+ if (ntfs_icx_parent_vcn(icx) == VCN_INDEX_ROOT_PARENT) {
+ ntfs_ir_leafify(icx, parent_ih);
+ goto ok;
+ }
+
+ ret = ntfs_index_rm_leaf(icx);
+ goto out;
+ }
+
+ if (ntfs_ih_reparent_end(icx, parent_ih, ib))
+ goto out;
+ok:
+ ret = STATUS_OK;
+out:
+ free(ib);
+ return ret;
+}
+
+static int ntfs_index_rm_node(ntfs_index_context *icx)
+{
+ int entry_pos, pindex;
+ VCN vcn;
+ INDEX_BLOCK *ib = NULL;
+ INDEX_ENTRY *ie_succ, *ie, *entry = icx->entry;
+ INDEX_HEADER *ih;
+ u32 new_size;
+ int delta, ret = STATUS_ERROR;
+
+ ntfs_log_trace("Entering\n");
+
+ if (!icx->ia_na) {
+ icx->ia_na = ntfs_ia_open(icx, icx->ni);
+ if (!icx->ia_na)
+ return STATUS_ERROR;
+ }
+
+ ib = ntfs_malloc(icx->block_size);
+ if (!ib)
+ return STATUS_ERROR;
+
+ ie_succ = ntfs_ie_get_next(icx->entry);
+ entry_pos = icx->parent_pos[icx->pindex]++;
+ pindex = icx->pindex;
+descend:
+ vcn = ntfs_ie_get_vcn(ie_succ);
+ if (ntfs_ib_read(icx, vcn, ib))
+ goto out;
+
+ ie_succ = ntfs_ie_get_first(&ib->index);
+
+ if (ntfs_icx_parent_inc(icx))
+ goto out;
+
+ icx->parent_vcn[icx->pindex] = vcn;
+ icx->parent_pos[icx->pindex] = 0;
+
+ if ((ib->index.ih_flags & NODE_MASK) == INDEX_NODE)
+ goto descend;
+
+ if (ntfs_ih_zero_entry(&ib->index)) {
+ errno = EIO;
+ ntfs_log_perror("Empty index block");
+ goto out;
+ }
+
+ ie = ntfs_ie_dup(ie_succ);
+ if (!ie)
+ goto out;
+
+ if (ntfs_ie_add_vcn(&ie))
+ goto out2;
+
+ ntfs_ie_set_vcn(ie, ntfs_ie_get_vcn(icx->entry));
+
+ if (icx->is_in_root)
+ ih = &icx->ir->index;
+ else
+ ih = &icx->ib->index;
+
+ delta = le16_to_cpu(ie->length) - le16_to_cpu(icx->entry->length);
+ new_size = le32_to_cpu(ih->index_length) + delta;
+ if (delta > 0) {
+ if (icx->is_in_root) {
+ ret = ntfs_ir_make_space(icx, new_size);
+ if (ret != STATUS_OK)
+ goto out2;
+
+ ih = &icx->ir->index;
+ entry = ntfs_ie_get_by_pos(ih, entry_pos);
+
+ } else if (new_size > le32_to_cpu(ih->allocated_size)) {
+ icx->pindex = pindex;
+ ret = ntfs_ib_split(icx, icx->ib);
+ if (ret == STATUS_OK)
+ ret = STATUS_KEEP_SEARCHING;
+ goto out2;
+ }
+ }
+
+ ntfs_ie_delete(ih, entry);
+ ntfs_ie_insert(ih, ie, entry);
+
+ if (icx->is_in_root) {
+ if (ntfs_ir_truncate(icx, new_size))
+ goto out2;
+ } else
+ if (ntfs_icx_ib_write(icx))
+ goto out2;
+
+ ntfs_ie_delete(&ib->index, ie_succ);
+
+ if (ntfs_ih_zero_entry(&ib->index)) {
+ if (ntfs_index_rm_leaf(icx))
+ goto out2;
+ } else
+ if (ntfs_ib_write(icx, ib))
+ goto out2;
+
+ ret = STATUS_OK;
+out2:
+ free(ie);
+out:
+ free(ib);
+ return ret;
+}
+
+/**
+ * ntfs_index_rm - remove entry from the index
+ * @icx: index context describing entry to delete
+ *
+ * Delete entry described by @icx from the index. Index context is always
+ * reinitialized after use of this function, so it can be used for index
+ * lookup once again.
+ *
+ * Return 0 on success or -1 on error with errno set to the error code.
+ */
+/*static JPA*/
+int ntfs_index_rm(ntfs_index_context *icx)
+{
+ INDEX_HEADER *ih;
+ int err, ret = STATUS_OK;
+
+ ntfs_log_trace("Entering\n");
+
+ if (!icx || (!icx->ib && !icx->ir) || ntfs_ie_end(icx->entry)) {
+ ntfs_log_error("Invalid arguments.\n");
+ errno = EINVAL;
+ goto err_out;
+ }
+ if (icx->is_in_root)
+ ih = &icx->ir->index;
+ else
+ ih = &icx->ib->index;
+
+ if (icx->entry->ie_flags & INDEX_ENTRY_NODE) {
+
+ ret = ntfs_index_rm_node(icx);
+
+ } else if (icx->is_in_root || !ntfs_ih_one_entry(ih)) {
+
+ ntfs_ie_delete(ih, icx->entry);
+
+ if (icx->is_in_root) {
+ err = ntfs_ir_truncate(icx, le32_to_cpu(ih->index_length));
+ if (err != STATUS_OK)
+ goto err_out;
+ } else
+ if (ntfs_icx_ib_write(icx))
+ goto err_out;
+ } else {
+ if (ntfs_index_rm_leaf(icx))
+ goto err_out;
+ }
+out:
+ return ret;
+err_out:
+ ret = STATUS_ERROR;
+ goto out;
+}
+
+int ntfs_index_remove(ntfs_inode *dir_ni, ntfs_inode *ni,
+ const void *key, const int keylen)
+{
+ int ret = STATUS_ERROR;
+ ntfs_index_context *icx;
+
+ icx = ntfs_index_ctx_get(dir_ni, NTFS_INDEX_I30, 4);
+ if (!icx)
+ return -1;
+
+ while (1) {
+
+ if (ntfs_index_lookup(key, keylen, icx))
+ goto err_out;
+
+ if ((((FILE_NAME_ATTR *)icx->data)->file_attributes &
+ FILE_ATTR_REPARSE_POINT)
+ && !ntfs_possible_symlink(ni)) {
+ errno = EOPNOTSUPP;
+ goto err_out;
+ }
+
+ ret = ntfs_index_rm(icx);
+ if (ret == STATUS_ERROR)
+ goto err_out;
+ else if (ret == STATUS_OK)
+ break;
+
+ ntfs_inode_mark_dirty(icx->actx->ntfs_ino);
+ ntfs_index_ctx_reinit(icx);
+ }
+
+ ntfs_inode_mark_dirty(icx->actx->ntfs_ino);
+out:
+ ntfs_index_ctx_put(icx);
+ return ret;
+err_out:
+ ret = STATUS_ERROR;
+ ntfs_log_perror("Delete failed");
+ goto out;
+}
+
+/**
+ * ntfs_index_root_get - read the index root of an attribute
+ * @ni: open ntfs inode in which the ntfs attribute resides
+ * @attr: attribute for which we want its index root
+ *
+ * This function will read the related index root an ntfs attribute.
+ *
+ * On success a buffer is allocated with the content of the index root
+ * and which needs to be freed when it's not needed anymore.
+ *
+ * On error NULL is returned with errno set to the error code.
+ */
+INDEX_ROOT *ntfs_index_root_get(ntfs_inode *ni, ATTR_RECORD *attr)
+{
+ ntfs_attr_search_ctx *ctx;
+ ntfschar *name;
+ INDEX_ROOT *root = NULL;
+
+ name = (ntfschar *)((u8 *)attr + le16_to_cpu(attr->name_offset));
+
+ if (!ntfs_ir_lookup(ni, name, attr->name_length, &ctx))
+ return NULL;
+
+ root = ntfs_malloc(sizeof(INDEX_ROOT));
+ if (!root)
+ goto out;
+
+ *root = *((INDEX_ROOT *)((u8 *)ctx->attr +
+ le16_to_cpu(ctx->attr->value_offset)));
+out:
+ ntfs_attr_put_search_ctx(ctx);
+ return root;
+}
+
+
+/*
+ * Walk down the index tree (leaf bound)
+ * until there are no subnode in the first index entry
+ * returns the entry at the bottom left in subnode
+ */
+
+static INDEX_ENTRY *ntfs_index_walk_down(INDEX_ENTRY *ie,
+ ntfs_index_context *ictx)
+{
+ INDEX_ENTRY *entry;
+ s64 vcn;
+
+ entry = ie;
+ do {
+ vcn = ntfs_ie_get_vcn(entry);
+ if (ictx->is_in_root) {
+
+ /* down from level zero */
+
+ ictx->ir = (INDEX_ROOT*)NULL;
+ ictx->ib = (INDEX_BLOCK*)ntfs_malloc(ictx->block_size);
+ ictx->pindex = 1;
+ ictx->is_in_root = FALSE;
+ } else {
+
+ /* down from non-zero level */
+
+ ictx->pindex++;
+ }
+ ictx->parent_pos[ictx->pindex] = 0;
+ ictx->parent_vcn[ictx->pindex] = vcn;
+ if (!ntfs_ib_read(ictx,vcn,ictx->ib)) {
+ ictx->entry = ntfs_ie_get_first(&ictx->ib->index);
+ entry = ictx->entry;
+ } else
+ entry = (INDEX_ENTRY*)NULL;
+ } while (entry && (entry->ie_flags & INDEX_ENTRY_NODE));
+ return (entry);
+}
+
+/*
+ * Walk up the index tree (root bound)
+ * until there is a valid data entry in parent
+ * returns the parent entry or NULL if no more parent
+ */
+
+static INDEX_ENTRY *ntfs_index_walk_up(INDEX_ENTRY *ie,
+ ntfs_index_context *ictx)
+{
+ INDEX_ENTRY *entry;
+ s64 vcn;
+
+ entry = ie;
+ if (ictx->pindex > 0) {
+ do {
+ ictx->pindex--;
+ if (!ictx->pindex) {
+
+ /* we have reached the root */
+
+ free(ictx->ib);
+ ictx->ib = (INDEX_BLOCK*)NULL;
+ ictx->is_in_root = TRUE;
+ /* a new search context is to be allocated */
+ if (ictx->actx)
+ free(ictx->actx);
+ ictx->ir = ntfs_ir_lookup(ictx->ni,
+ ictx->name, ictx->name_len,
+ &ictx->actx);
+ if (ictx->ir)
+ entry = ntfs_ie_get_by_pos(
+ &ictx->ir->index,
+ ictx->parent_pos[ictx->pindex]);
+ else
+ entry = (INDEX_ENTRY*)NULL;
+ } else {
+ /* up into non-root node */
+ vcn = ictx->parent_vcn[ictx->pindex];
+ if (!ntfs_ib_read(ictx,vcn,ictx->ib)) {
+ entry = ntfs_ie_get_by_pos(
+ &ictx->ib->index,
+ ictx->parent_pos[ictx->pindex]);
+ } else
+ entry = (INDEX_ENTRY*)NULL;
+ }
+ ictx->entry = entry;
+ } while (entry && (ictx->pindex > 0)
+ && (entry->ie_flags & INDEX_ENTRY_END));
+ } else
+ entry = (INDEX_ENTRY*)NULL;
+ return (entry);
+}
+
+/*
+ * Get next entry in an index according to collating sequence.
+ * Must be initialized through a ntfs_index_lookup()
+ *
+ * Returns next entry or NULL if none
+ *
+ * Sample layout :
+ *
+ * +---+---+---+---+---+---+---+---+ n ptrs to subnodes
+ * | | | 10| 25| 33| | | | n-1 keys in between
+ * +---+---+---+---+---+---+---+---+ no key in last entry
+ * | A | A
+ * | | | +-------------------------------+
+ * +--------------------------+ | +-----+ |
+ * | +--+ | |
+ * V | V |
+ * +---+---+---+---+---+---+---+---+ | +---+---+---+---+---+---+---+---+
+ * | 11| 12| 13| 14| 15| 16| 17| | | | 26| 27| 28| 29| 30| 31| 32| |
+ * +---+---+---+---+---+---+---+---+ | +---+---+---+---+---+---+---+---+
+ * | |
+ * +-----------------------+ |
+ * | |
+ * +---+---+---+---+---+---+---+---+
+ * | 18| 19| 20| 21| 22| 23| 24| |
+ * +---+---+---+---+---+---+---+---+
+ */
+
+INDEX_ENTRY *ntfs_index_next(INDEX_ENTRY *ie, ntfs_index_context *ictx)
+{
+ INDEX_ENTRY *next;
+ int flags;
+
+ /*
+ * lookup() may have returned an invalid node
+ * when searching for a partial key
+ * if this happens, walk up
+ */
+
+ if (ie->ie_flags & INDEX_ENTRY_END)
+ next = ntfs_index_walk_up(ie, ictx);
+ else {
+ /*
+ * get next entry in same node
+ * there is always one after any entry with data
+ */
+
+ next = (INDEX_ENTRY*)((char*)ie + le16_to_cpu(ie->length));
+ ++ictx->parent_pos[ictx->pindex];
+ flags = next->ie_flags;
+
+ /* walk down if it has a subnode */
+
+ if (flags & INDEX_ENTRY_NODE) {
+ next = ntfs_index_walk_down(next,ictx);
+ } else {
+
+ /* walk up it has no subnode, nor data */
+
+ if (flags & INDEX_ENTRY_END) {
+ next = ntfs_index_walk_up(next, ictx);
+ }
+ }
+ }
+ /* return NULL if stuck at end of a block */
+
+ if (next && (next->ie_flags & INDEX_ENTRY_END))
+ next = (INDEX_ENTRY*)NULL;
+ return (next);
+}
+
+