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/cache.c b/libntfs-3g/cache.c
new file mode 100755
index 0000000..dd14767
--- a/dev/null
+++ b/libntfs-3g/cache.c
@@ -0,0 +1,609 @@
+/**
+ * cache.c : deal with LRU caches
+ *
+ * Copyright (c) 2008-2009 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
+
+#include "types.h"
+#include "security.h"
+#include "cache.h"
+#include "misc.h"
+#include "logging.h"
+
+/*
+ * General functions to deal with LRU caches
+ *
+ * The cached data have to be organized in a structure in which
+ * the first fields must follow a mandatory pattern and further
+ * fields may contain any fixed size data. They are stored in an
+ * LRU list.
+ *
+ * A compare function must be provided for finding a wanted entry
+ * in the cache. Another function may be provided for invalidating
+ * an entry to facilitate multiple invalidation.
+ *
+ * These functions never return error codes. When there is a
+ * shortage of memory, data is simply not cached.
+ * When there is a hashing bug, hashing is dropped, and sequential
+ * searches are used.
+ */
+
+/*
+ * Enter a new hash index, after a new record has been inserted
+ *
+ * Do not call when a record has been modified (with no key change)
+ */
+
+static void inserthashindex(struct CACHE_HEADER *cache,
+ struct CACHED_GENERIC *current)
+{
+ int h;
+ struct HASH_ENTRY *link;
+ struct HASH_ENTRY *first;
+
+ if (cache->dohash) {
+ h = cache->dohash(current);
+ if ((h >= 0) && (h < cache->max_hash)) {
+ /* get a free link and insert at top of hash list */
+ link = cache->free_hash;
+ if (link) {
+ cache->free_hash = link->next;
+ first = cache->first_hash[h];
+ if (first)
+ link->next = first;
+ else
+ link->next = NULL;
+ link->entry = current;
+ cache->first_hash[h] = link;
+ } else {
+ ntfs_log_error("No more hash entries,"
+ " cache %s hashing dropped\n",
+ cache->name);
+ cache->dohash = (cache_hash)NULL;
+ }
+ } else {
+ ntfs_log_error("Illegal hash value,"
+ " cache %s hashing dropped\n",
+ cache->name);
+ cache->dohash = (cache_hash)NULL;
+ }
+ }
+}
+
+/*
+ * Drop a hash index when a record is about to be deleted
+ */
+
+static void drophashindex(struct CACHE_HEADER *cache,
+ const struct CACHED_GENERIC *current, int hash)
+{
+ struct HASH_ENTRY *link;
+ struct HASH_ENTRY *previous;
+
+ if (cache->dohash) {
+ if ((hash >= 0) && (hash < cache->max_hash)) {
+ /* find the link and unlink */
+ link = cache->first_hash[hash];
+ previous = (struct HASH_ENTRY*)NULL;
+ while (link && (link->entry != current)) {
+ previous = link;
+ link = link->next;
+ }
+ if (link) {
+ if (previous)
+ previous->next = link->next;
+ else
+ cache->first_hash[hash] = link->next;
+ link->next = cache->free_hash;
+ cache->free_hash = link;
+ } else {
+ ntfs_log_error("Bad hash list,"
+ " cache %s hashing dropped\n",
+ cache->name);
+ cache->dohash = (cache_hash)NULL;
+ }
+ } else {
+ ntfs_log_error("Illegal hash value,"
+ " cache %s hashing dropped\n",
+ cache->name);
+ cache->dohash = (cache_hash)NULL;
+ }
+ }
+}
+
+/*
+ * Fetch an entry from cache
+ *
+ * returns the cache entry, or NULL if not available
+ * The returned entry may be modified, but not freed
+ */
+
+struct CACHED_GENERIC *ntfs_fetch_cache(struct CACHE_HEADER *cache,
+ const struct CACHED_GENERIC *wanted, cache_compare compare)
+{
+ struct CACHED_GENERIC *current;
+ struct CACHED_GENERIC *previous;
+ struct HASH_ENTRY *link;
+ int h;
+
+ current = (struct CACHED_GENERIC*)NULL;
+ if (cache) {
+ if (cache->dohash) {
+ /*
+ * When possible, use the hash table to
+ * locate the entry if present
+ */
+ h = cache->dohash(wanted);
+ link = cache->first_hash[h];
+ while (link && compare(link->entry, wanted))
+ link = link->next;
+ if (link)
+ current = link->entry;
+ }
+ if (!cache->dohash) {
+ /*
+ * Search sequentially in LRU list if no hash table
+ * or if hashing has just failed
+ */
+ current = cache->most_recent_entry;
+ while (current
+ && compare(current, wanted)) {
+ current = current->next;
+ }
+ }
+ if (current) {
+ previous = current->previous;
+ cache->hits++;
+ if (previous) {
+ /*
+ * found and not at head of list, unlink from current
+ * position and relink as head of list
+ */
+ previous->next = current->next;
+ if (current->next)
+ current->next->previous
+ = current->previous;
+ else
+ cache->oldest_entry
+ = current->previous;
+ current->next = cache->most_recent_entry;
+ current->previous
+ = (struct CACHED_GENERIC*)NULL;
+ cache->most_recent_entry->previous = current;
+ cache->most_recent_entry = current;
+ }
+ }
+ cache->reads++;
+ }
+ return (current);
+}
+
+/*
+ * Enter an inode number into cache
+ * returns the cache entry or NULL if not possible
+ */
+
+struct CACHED_GENERIC *ntfs_enter_cache(struct CACHE_HEADER *cache,
+ const struct CACHED_GENERIC *item,
+ cache_compare compare)
+{
+ struct CACHED_GENERIC *current;
+ struct CACHED_GENERIC *before;
+ struct HASH_ENTRY *link;
+ int h;
+
+ current = (struct CACHED_GENERIC*)NULL;
+ if (cache) {
+ if (cache->dohash) {
+ /*
+ * When possible, use the hash table to
+ * find out whether the entry if present
+ */
+ h = cache->dohash(item);
+ link = cache->first_hash[h];
+ while (link && compare(link->entry, item))
+ link = link->next;
+ if (link) {
+ current = link->entry;
+ }
+ }
+ if (!cache->dohash) {
+ /*
+ * Search sequentially in LRU list to locate the end,
+ * and find out whether the entry is already in list
+ * As we normally go to the end, no statistics is
+ * kept.
+ */
+ current = cache->most_recent_entry;
+ while (current
+ && compare(current, item)) {
+ current = current->next;
+ }
+ }
+
+ if (!current) {
+ /*
+ * Not in list, get a free entry or reuse the
+ * last entry, and relink as head of list
+ * Note : we assume at least three entries, so
+ * before, previous and first are different when
+ * an entry is reused.
+ */
+
+ if (cache->free_entry) {
+ current = cache->free_entry;
+ cache->free_entry = cache->free_entry->next;
+ if (item->varsize) {
+ current->variable = ntfs_malloc(
+ item->varsize);
+ } else
+ current->variable = (void*)NULL;
+ current->varsize = item->varsize;
+ if (!cache->oldest_entry)
+ cache->oldest_entry = current;
+ } else {
+ /* reusing the oldest entry */
+ current = cache->oldest_entry;
+ before = current->previous;
+ before->next = (struct CACHED_GENERIC*)NULL;
+ if (cache->dohash)
+ drophashindex(cache,current,
+ cache->dohash(current));
+ if (cache->dofree)
+ cache->dofree(current);
+ cache->oldest_entry = current->previous;
+ if (item->varsize) {
+ if (current->varsize)
+ current->variable = realloc(
+ current->variable,
+ item->varsize);
+ else
+ current->variable = ntfs_malloc(
+ item->varsize);
+ } else {
+ if (current->varsize)
+ free(current->variable);
+ current->variable = (void*)NULL;
+ }
+ current->varsize = item->varsize;
+ }
+ current->next = cache->most_recent_entry;
+ current->previous = (struct CACHED_GENERIC*)NULL;
+ if (cache->most_recent_entry)
+ cache->most_recent_entry->previous = current;
+ cache->most_recent_entry = current;
+ memcpy(current->fixed, item->fixed, cache->fixed_size);
+ if (item->varsize) {
+ if (current->variable) {
+ memcpy(current->variable,
+ item->variable, item->varsize);
+ } else {
+ /*
+ * no more memory for variable part
+ * recycle entry in free list
+ * not an error, just uncacheable
+ */
+ cache->most_recent_entry = current->next;
+ current->next = cache->free_entry;
+ cache->free_entry = current;
+ current = (struct CACHED_GENERIC*)NULL;
+ }
+ } else {
+ current->variable = (void*)NULL;
+ current->varsize = 0;
+ }
+ if (cache->dohash && current)
+ inserthashindex(cache,current);
+ }
+ cache->writes++;
+ }
+ return (current);
+}
+
+/*
+ * Invalidate a cache entry
+ * The entry is moved to the free entry list
+ * A specific function may be called for entry deletion
+ */
+
+static void do_invalidate(struct CACHE_HEADER *cache,
+ struct CACHED_GENERIC *current, int flags)
+{
+ struct CACHED_GENERIC *previous;
+
+ previous = current->previous;
+ if ((flags & CACHE_FREE) && cache->dofree)
+ cache->dofree(current);
+ /*
+ * Relink into free list
+ */
+ if (current->next)
+ current->next->previous = current->previous;
+ else
+ cache->oldest_entry = current->previous;
+ if (previous)
+ previous->next = current->next;
+ else
+ cache->most_recent_entry = current->next;
+ current->next = cache->free_entry;
+ cache->free_entry = current;
+ if (current->variable)
+ free(current->variable);
+ current->varsize = 0;
+ }
+
+
+/*
+ * Invalidate entries in cache
+ *
+ * Several entries may have to be invalidated (at least for inodes
+ * associated to directories which have been renamed), a different
+ * compare function may be provided to select entries to invalidate
+ *
+ * Returns the number of deleted entries, this can be used by
+ * the caller to signal a cache corruption if the entry was
+ * supposed to be found.
+ */
+
+int ntfs_invalidate_cache(struct CACHE_HEADER *cache,
+ const struct CACHED_GENERIC *item, cache_compare compare,
+ int flags)
+{
+ struct CACHED_GENERIC *current;
+ struct CACHED_GENERIC *previous;
+ struct CACHED_GENERIC *next;
+ struct HASH_ENTRY *link;
+ int count;
+ int h;
+
+ current = (struct CACHED_GENERIC*)NULL;
+ count = 0;
+ if (cache) {
+ if (!(flags & CACHE_NOHASH) && cache->dohash) {
+ /*
+ * When possible, use the hash table to
+ * find out whether the entry if present
+ */
+ h = cache->dohash(item);
+ link = cache->first_hash[h];
+ while (link) {
+ if (compare(link->entry, item))
+ link = link->next;
+ else {
+ current = link->entry;
+ link = link->next;
+ if (current) {
+ drophashindex(cache,current,h);
+ do_invalidate(cache,
+ current,flags);
+ count++;
+ }
+ }
+ }
+ }
+ if ((flags & CACHE_NOHASH) || !cache->dohash) {
+ /*
+ * Search sequentially in LRU list
+ */
+ current = cache->most_recent_entry;
+ previous = (struct CACHED_GENERIC*)NULL;
+ while (current) {
+ if (!compare(current, item)) {
+ next = current->next;
+ if (cache->dohash)
+ drophashindex(cache,current,
+ cache->dohash(current));
+ do_invalidate(cache,current,flags);
+ current = next;
+ count++;
+ } else {
+ previous = current;
+ current = current->next;
+ }
+ }
+ }
+ }
+ return (count);
+}
+
+int ntfs_remove_cache(struct CACHE_HEADER *cache,
+ struct CACHED_GENERIC *item, int flags)
+{
+ int count;
+
+ count = 0;
+ if (cache) {
+ if (cache->dohash)
+ drophashindex(cache,item,cache->dohash(item));
+ do_invalidate(cache,item,flags);
+ count++;
+ }
+ return (count);
+}
+
+/*
+ * Free memory allocated to a cache
+ */
+
+static void ntfs_free_cache(struct CACHE_HEADER *cache)
+{
+ struct CACHED_GENERIC *entry;
+
+ if (cache) {
+ for (entry=cache->most_recent_entry; entry; entry=entry->next) {
+ if (cache->dofree)
+ cache->dofree(entry);
+ if (entry->variable)
+ free(entry->variable);
+ }
+ free(cache);
+ }
+}
+
+/*
+ * Create a cache
+ *
+ * Returns the cache header, or NULL if the cache could not be created
+ */
+
+static struct CACHE_HEADER *ntfs_create_cache(const char *name,
+ cache_free dofree, cache_hash dohash,
+ int full_item_size,
+ int item_count, int max_hash)
+{
+ struct CACHE_HEADER *cache;
+ struct CACHED_GENERIC *pc;
+ struct CACHED_GENERIC *qc;
+ struct HASH_ENTRY *ph;
+ struct HASH_ENTRY *qh;
+ struct HASH_ENTRY **px;
+ size_t size;
+ int i;
+
+ size = sizeof(struct CACHE_HEADER) + item_count*full_item_size;
+ if (max_hash)
+ size += item_count*sizeof(struct HASH_ENTRY)
+ + max_hash*sizeof(struct HASH_ENTRY*);
+ cache = (struct CACHE_HEADER*)ntfs_malloc(size);
+ if (cache) {
+ /* header */
+ cache->name = name;
+ cache->dofree = dofree;
+ if (dohash && max_hash) {
+ cache->dohash = dohash;
+ cache->max_hash = max_hash;
+ } else {
+ cache->dohash = (cache_hash)NULL;
+ cache->max_hash = 0;
+ }
+ cache->fixed_size = full_item_size - sizeof(struct CACHED_GENERIC);
+ cache->reads = 0;
+ cache->writes = 0;
+ cache->hits = 0;
+ /* chain the data entries, and mark an invalid entry */
+ cache->most_recent_entry = (struct CACHED_GENERIC*)NULL;
+ cache->oldest_entry = (struct CACHED_GENERIC*)NULL;
+ cache->free_entry = &cache->entry[0];
+ pc = &cache->entry[0];
+ for (i=0; i<(item_count - 1); i++) {
+ qc = (struct CACHED_GENERIC*)((char*)pc
+ + full_item_size);
+ pc->next = qc;
+ pc->variable = (void*)NULL;
+ pc->varsize = 0;
+ pc = qc;
+ }
+ /* special for the last entry */
+ pc->next = (struct CACHED_GENERIC*)NULL;
+ pc->variable = (void*)NULL;
+ pc->varsize = 0;
+
+ if (max_hash) {
+ /* chain the hash entries */
+ ph = (struct HASH_ENTRY*)(((char*)pc) + full_item_size);
+ cache->free_hash = ph;
+ for (i=0; i<(item_count - 1); i++) {
+ qh = &ph[1];
+ ph->next = qh;
+ ph = qh;
+ }
+ /* special for the last entry */
+ if (item_count) {
+ ph->next = (struct HASH_ENTRY*)NULL;
+ }
+ /* create and initialize the hash indexes */
+ px = (struct HASH_ENTRY**)&ph[1];
+ cache->first_hash = px;
+ for (i=0; i<max_hash; i++)
+ px[i] = (struct HASH_ENTRY*)NULL;
+ } else {
+ cache->free_hash = (struct HASH_ENTRY*)NULL;
+ cache->first_hash = (struct HASH_ENTRY**)NULL;
+ }
+ }
+ return (cache);
+}
+
+/*
+ * Create all LRU caches
+ *
+ * No error return, if creation is not possible, cacheing will
+ * just be not available
+ */
+
+void ntfs_create_lru_caches(ntfs_volume *vol)
+{
+#if CACHE_INODE_SIZE
+ /* inode cache */
+ vol->xinode_cache = ntfs_create_cache("inode",(cache_free)NULL,
+ ntfs_dir_inode_hash, sizeof(struct CACHED_INODE),
+ CACHE_INODE_SIZE, 2*CACHE_INODE_SIZE);
+#endif
+#if CACHE_NIDATA_SIZE
+ /* idata cache */
+ vol->nidata_cache = ntfs_create_cache("nidata",
+ ntfs_inode_nidata_free, ntfs_inode_nidata_hash,
+ sizeof(struct CACHED_NIDATA),
+ CACHE_NIDATA_SIZE, 2*CACHE_NIDATA_SIZE);
+#endif
+#if CACHE_LOOKUP_SIZE
+ /* lookup cache */
+ vol->lookup_cache = ntfs_create_cache("lookup",
+ (cache_free)NULL, ntfs_dir_lookup_hash,
+ sizeof(struct CACHED_LOOKUP),
+ CACHE_LOOKUP_SIZE, 2*CACHE_LOOKUP_SIZE);
+#endif
+ vol->securid_cache = ntfs_create_cache("securid",(cache_free)NULL,
+ (cache_hash)NULL,sizeof(struct CACHED_SECURID), CACHE_SECURID_SIZE, 0);
+#if CACHE_LEGACY_SIZE
+ vol->legacy_cache = ntfs_create_cache("legacy",(cache_free)NULL,
+ (cache_hash)NULL, sizeof(struct CACHED_PERMISSIONS_LEGACY), CACHE_LEGACY_SIZE, 0);
+#endif
+}
+
+/*
+ * Free all LRU caches
+ */
+
+void ntfs_free_lru_caches(ntfs_volume *vol)
+{
+#if CACHE_INODE_SIZE
+ ntfs_free_cache(vol->xinode_cache);
+#endif
+#if CACHE_NIDATA_SIZE
+ ntfs_free_cache(vol->nidata_cache);
+#endif
+#if CACHE_LOOKUP_SIZE
+ ntfs_free_cache(vol->lookup_cache);
+#endif
+ ntfs_free_cache(vol->securid_cache);
+#if CACHE_LEGACY_SIZE
+ ntfs_free_cache(vol->legacy_cache);
+#endif
+}