Commit 185decf4 authored by Anton Altaparmakov's avatar Anton Altaparmakov

NTFS: 2.1.2 release: Fix buggy free cluster and free inode determination logic.

parent 3880b6d5
...@@ -247,6 +247,13 @@ ChangeLog ...@@ -247,6 +247,13 @@ ChangeLog
Note, a technical ChangeLog aimed at kernel hackers is in fs/ntfs/ChangeLog. Note, a technical ChangeLog aimed at kernel hackers is in fs/ntfs/ChangeLog.
2.1.2:
- Major bug fixes aleviating the hangs in statfs experienced by some
users.
2.1.1:
- Update handling of compressed files so people no longer get the
frequently reported warning messages about initialized_size !=
data_size.
2.1.0: 2.1.0:
- Add configuration option for developmental write support. - Add configuration option for developmental write support.
- Initial implementation of file overwriting. (Writes to resident files - Initial implementation of file overwriting. (Writes to resident files
......
...@@ -20,11 +20,16 @@ ToDo: ...@@ -20,11 +20,16 @@ ToDo:
sufficient for synchronisation here. We then just need to make sure sufficient for synchronisation here. We then just need to make sure
ntfs_readpage/writepage/truncate interoperate properly with us. ntfs_readpage/writepage/truncate interoperate properly with us.
2.1.1 - WIP 2.1.2 - Important bug fixes aleviating the hangs in statfs.
- Fix buggy free cluster and free inode determination logic.
2.1.1 - Important bug fix aleviating the random hangs.
- Add handling for initialized_size != data_size in compressed files. - Add handling for initialized_size != data_size in compressed files.
- Reduce function local stack usage from 0x3d4 bytes to just noise in - Reduce function local stack usage from 0x3d4 bytes to just noise in
fs/ntfs/upcase.c. (Randy Dunlap <rddunlap@osdl.ord>) fs/ntfs/upcase.c. (Randy Dunlap <rddunlap@osdl.ord>)
- Remove compiler warnings for newer gcc.
2.1.0 - First steps towards write support: implement file overwrite. 2.1.0 - First steps towards write support: implement file overwrite.
......
...@@ -5,7 +5,7 @@ obj-$(CONFIG_NTFS_FS) += ntfs.o ...@@ -5,7 +5,7 @@ obj-$(CONFIG_NTFS_FS) += ntfs.o
ntfs-objs := aops.o attrib.o compress.o debug.o dir.o file.o inode.o mft.o \ ntfs-objs := aops.o attrib.o compress.o debug.o dir.o file.o inode.o mft.o \
mst.o namei.o super.o sysctl.o time.o unistr.o upcase.o mst.o namei.o super.o sysctl.o time.o unistr.o upcase.o
EXTRA_CFLAGS = -DNTFS_VERSION=\"2.1.1-WIP\" EXTRA_CFLAGS = -DNTFS_VERSION=\"2.1.2\"
ifeq ($(CONFIG_NTFS_DEBUG),y) ifeq ($(CONFIG_NTFS_DEBUG),y)
EXTRA_CFLAGS += -DDEBUG EXTRA_CFLAGS += -DDEBUG
......
...@@ -1060,78 +1060,93 @@ static void ntfs_put_super(struct super_block *vfs_sb) ...@@ -1060,78 +1060,93 @@ static void ntfs_put_super(struct super_block *vfs_sb)
* get_nr_free_clusters - return the number of free clusters on a volume * get_nr_free_clusters - return the number of free clusters on a volume
* @vol: ntfs volume for which to obtain free cluster count * @vol: ntfs volume for which to obtain free cluster count
* *
* Calculate the number of free clusters on the mounted NTFS volume @vol. * Calculate the number of free clusters on the mounted NTFS volume @vol. We
* actually calculate the number of clusters in use instead because this
* allows us to not care about partial pages as these will be just zero filled
* and hence not be counted as allocated clusters.
* *
* Errors are ignored and we just return the number of free clusters we have * The only particularity is that clusters beyond the end of the logical ntfs
* found. This means we return an underestimate on error. * volume will be marked as allocated to prevent errors which means we have to
* discount those at the end. This is important as the cluster bitmap always
* has a size in multiples of 8 bytes, i.e. up to 63 clusters could be outside
* the logical volume and marked in use when they are not as they do not exist.
*
* If any pages cannot be read we assume all clusters in the erroring pages are
* in use. This means we return an underestimate on errors which is better than
* an overestimate.
*/ */
static s64 get_nr_free_clusters(ntfs_volume *vol) static s64 get_nr_free_clusters(ntfs_volume *vol)
{ {
s64 nr_free = vol->nr_clusters;
u32 *kaddr;
struct address_space *mapping = vol->lcnbmp_ino->i_mapping; struct address_space *mapping = vol->lcnbmp_ino->i_mapping;
filler_t *readpage = (filler_t*)mapping->a_ops->readpage; filler_t *readpage = (filler_t*)mapping->a_ops->readpage;
struct page *page; struct page *page;
unsigned long index, max_index; unsigned long index, max_index;
unsigned int max_size, i; unsigned int max_size;
s64 nr_free = 0LL;
u32 *b;
ntfs_debug("Entering."); ntfs_debug("Entering.");
/* Serialize accesses to the cluster bitmap. */ /* Serialize accesses to the cluster bitmap. */
down_read(&vol->lcnbmp_lock); down_read(&vol->lcnbmp_lock);
/* /*
* Convert the number of bits into bytes rounded up, then convert into * Convert the number of bits into bytes rounded up, then convert into
* multiples of PAGE_CACHE_SIZE. * multiples of PAGE_CACHE_SIZE, rounding up so that if we have one
* full and one partial page max_index = 2.
*/ */
max_index = (vol->nr_clusters + 7) >> (3 + PAGE_CACHE_SHIFT); max_index = (((vol->nr_clusters + 7) >> 3) + PAGE_CACHE_SIZE - 1) >>
PAGE_CACHE_SHIFT;
/* Use multiples of 4 bytes. */ /* Use multiples of 4 bytes. */
max_size = PAGE_CACHE_SIZE >> 2; max_size = PAGE_CACHE_SIZE >> 2;
ntfs_debug("Reading $BITMAP, max_index = 0x%lx, max_size = 0x%x.", ntfs_debug("Reading $Bitmap, max_index = 0x%lx, max_size = 0x%x.",
max_index, max_size); max_index, max_size);
for (index = 0UL; index < max_index;) { for (index = 0UL; index < max_index; index++) {
handle_partial_page: unsigned int i;
/* /*
* Read the page from page cache, getting it from backing store * Read the page from page cache, getting it from backing store
* if necessary, and increment the use count. * if necessary, and increment the use count.
*/ */
page = read_cache_page(mapping, index++, (filler_t*)readpage, page = read_cache_page(mapping, index, (filler_t*)readpage,
NULL); NULL);
/* Ignore pages which errored synchronously. */ /* Ignore pages which errored synchronously. */
if (IS_ERR(page)) { if (IS_ERR(page)) {
ntfs_debug("Sync read_cache_page() error. Skipping " ntfs_debug("Sync read_cache_page() error. Skipping "
"page (index 0x%lx).", index - 1); "page (index 0x%lx).", index);
nr_free -= PAGE_CACHE_SIZE * 8;
continue; continue;
} }
wait_on_page_locked(page); wait_on_page_locked(page);
/* Ignore pages which errored asynchronously. */
if (!PageUptodate(page)) { if (!PageUptodate(page)) {
ntfs_debug("Async read_cache_page() error. Skipping " ntfs_debug("Async read_cache_page() error. Skipping "
"page (index 0x%lx).", index - 1); "page (index 0x%lx).", index);
/* Ignore pages which errored asynchronously. */
page_cache_release(page); page_cache_release(page);
nr_free -= PAGE_CACHE_SIZE * 8;
continue; continue;
} }
b = (u32*)kmap(page); kaddr = (u32*)kmap_atomic(page, KM_USER0);
/* For each 4 bytes, add up the number zero bits. */
for (i = 0; i < max_size; i++)
nr_free += (s64)(32 - hweight32(b[i]));
kunmap(page);
page_cache_release(page);
}
if (max_size == PAGE_CACHE_SIZE >> 2) {
/* /*
* Get the multiples of 4 bytes in use in the final partial * For each 4 bytes, subtract the number of set bits. If this
* page. * is the last page and it is partial we don't really care as
* it just means we do a little extra work but it won't affect
* the result as all out of range bytes are set to zero by
* ntfs_readpage().
*/ */
max_size = ((((vol->nr_clusters + 7) >> 3) & ~PAGE_CACHE_MASK) for (i = 0; i < max_size; i++)
+ 3) >> 2; nr_free -= (s64)hweight32(kaddr[i]);
/* If there is a partial page go back and do it. */ kunmap_atomic(kaddr, KM_USER0);
if (max_size) { page_cache_release(page);
ntfs_debug("Handling partial page, max_size = 0x%x.",
max_size);
goto handle_partial_page;
}
} }
ntfs_debug("Finished reading $BITMAP, last index = 0x%lx", index - 1); ntfs_debug("Finished reading $Bitmap, last index = 0x%lx.", index - 1);
/*
* Fixup for eventual bits outside logical ntfs volume (see function
* description above).
*/
if (vol->nr_clusters & 63)
nr_free += 64 - (vol->nr_clusters & 63);
up_read(&vol->lcnbmp_lock); up_read(&vol->lcnbmp_lock);
/* If errors occured we may well have gone below zero, fix this. */
if (nr_free < 0)
nr_free = 0;
ntfs_debug("Exiting."); ntfs_debug("Exiting.");
return nr_free; return nr_free;
} }
...@@ -1141,64 +1156,81 @@ static s64 get_nr_free_clusters(ntfs_volume *vol) ...@@ -1141,64 +1156,81 @@ static s64 get_nr_free_clusters(ntfs_volume *vol)
* @vol: ntfs volume for which to obtain free inode count * @vol: ntfs volume for which to obtain free inode count
* *
* Calculate the number of free mft records (inodes) on the mounted NTFS * Calculate the number of free mft records (inodes) on the mounted NTFS
* volume @vol. * volume @vol. We actually calculate the number of mft records in use instead
* because this allows us to not care about partial pages as these will be just
* zero filled and hence not be counted as allocated mft record.
* *
* Errors are ignored and we just return the number of free inodes we have * If any pages cannot be read we assume all mft records in the erroring pages
* found. This means we return an underestimate on error. * are in use. This means we return an underestimate on errors which is better
* than an overestimate.
* *
* NOTE: Caller must hold mftbmp_lock rw_semaphore for reading or writing. * NOTE: Caller must hold mftbmp_lock rw_semaphore for reading or writing.
*/ */
static unsigned long __get_nr_free_mft_records(ntfs_volume *vol) static unsigned long __get_nr_free_mft_records(ntfs_volume *vol)
{ {
struct address_space *mapping; s64 nr_free = vol->nr_mft_records;
u32 *kaddr;
struct address_space *mapping = vol->mftbmp_ino->i_mapping;
filler_t *readpage = (filler_t*)mapping->a_ops->readpage;
struct page *page; struct page *page;
unsigned long index, max_index, nr_free = 0; unsigned long index, max_index;
unsigned int max_size, i; unsigned int max_size;
u32 *b;
mapping = vol->mftbmp_ino->i_mapping; ntfs_debug("Entering.");
/* /*
* Convert the number of bits into bytes rounded up to a multiple of 8 * Convert the number of bits into bytes rounded up, then convert into
* bytes, then convert into multiples of PAGE_CACHE_SIZE. * multiples of PAGE_CACHE_SIZE, rounding up so that if we have one
* full and one partial page max_index = 2.
*/ */
max_index = (((vol->nr_mft_records + 7) >> 3) + 7) >> PAGE_CACHE_SHIFT; max_index = (((vol->nr_mft_records + 7) >> 3) + PAGE_CACHE_SIZE - 1) >>
PAGE_CACHE_SHIFT;
/* Use multiples of 4 bytes. */ /* Use multiples of 4 bytes. */
max_size = PAGE_CACHE_SIZE >> 2; max_size = PAGE_CACHE_SIZE >> 2;
ntfs_debug("Reading $MFT/$BITMAP, max_index = 0x%lx, max_size = " ntfs_debug("Reading $MFT/$BITMAP, max_index = 0x%lx, max_size = "
"0x%x.", max_index, max_size); "0x%x.", max_index, max_size);
for (index = 0UL; index < max_index;) { for (index = 0UL; index < max_index; index++) {
handle_partial_page: unsigned int i;
page = ntfs_map_page(mapping, index++); /*
* Read the page from page cache, getting it from backing store
* if necessary, and increment the use count.
*/
page = read_cache_page(mapping, index, (filler_t*)readpage,
NULL);
/* Ignore pages which errored synchronously. */
if (IS_ERR(page)) { if (IS_ERR(page)) {
ntfs_debug("ntfs_map_page() error. Skipping page " ntfs_debug("Sync read_cache_page() error. Skipping "
"(index 0x%lx).", index - 1); "page (index 0x%lx).", index);
nr_free -= PAGE_CACHE_SIZE * 8;
continue; continue;
} }
b = (u32*)page_address(page); wait_on_page(page);
/* For each 4 bytes, add up the number of zero bits. */ /* Ignore pages which errored asynchronously. */
for (i = 0; i < max_size; i++) if (!PageUptodate(page)) {
nr_free += 32 - hweight32(b[i]); ntfs_debug("Async read_cache_page() error. Skipping "
ntfs_unmap_page(page); "page (index 0x%lx).", index);
} page_cache_release(page);
if (index == max_index) { nr_free -= PAGE_CACHE_SIZE * 8;
continue;
}
kaddr = (u32*)kmap_atomic(page, KM_USER0);
/* /*
* Get the multiples of 4 bytes in use in the final partial * For each 4 bytes, subtract the number of set bits. If this
* page. * is the last page and it is partial we don't really care as
* it just means we do a little extra work but it won't affect
* the result as all out of range bytes are set to zero by
* ntfs_readpage().
*/ */
max_size = ((((((vol->nr_mft_records + 7) >> 3) + 7) & ~7) & for (i = 0; i < max_size; i++)
~PAGE_CACHE_MASK) + 3) >> 2; nr_free -= (s64)hweight32(kaddr[i]);
/* If there is a partial page go back and do it. */ kunmap_atomic(kaddr, KM_USER0);
if (max_size) { page_cache_release(page);
/* Compensate for out of bounds zero bits. */
if ((i = vol->nr_mft_records & 31))
nr_free -= 32 - i;
ntfs_debug("Handling partial page, max_size = 0x%x",
max_size);
goto handle_partial_page;
}
} }
ntfs_debug("Finished reading $MFT/$BITMAP, last index = 0x%lx", ntfs_debug("Finished reading $MFT/$BITMAP, last index = 0x%lx.",
index - 1); index - 1);
/* If errors occured we may well have gone below zero, fix this. */
if (nr_free < 0)
nr_free = 0;
ntfs_debug("Exiting.");
return nr_free; return nr_free;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment