Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
185decf4
Commit
185decf4
authored
Mar 13, 2003
by
Anton Altaparmakov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
NTFS: 2.1.2 release: Fix buggy free cluster and free inode determination logic.
parent
3880b6d5
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
117 additions
and
73 deletions
+117
-73
Documentation/filesystems/ntfs.txt
Documentation/filesystems/ntfs.txt
+7
-0
fs/ntfs/ChangeLog
fs/ntfs/ChangeLog
+6
-1
fs/ntfs/Makefile
fs/ntfs/Makefile
+1
-1
fs/ntfs/super.c
fs/ntfs/super.c
+103
-71
No files found.
Documentation/filesystems/ntfs.txt
View file @
185decf4
...
@@ -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
...
...
fs/ntfs/ChangeLog
View file @
185decf4
...
@@ -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.
...
...
fs/ntfs/Makefile
View file @
185decf4
...
@@ -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
...
...
fs/ntfs/super.c
View file @
185decf4
...
@@ -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 $B
ITMAP
, max_index = 0x%lx, max_size = 0x%x."
,
ntfs_debug
(
"Reading $B
itmap
, 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 each 4 bytes, subtract the number of set bits. If this
* 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().
*/
for
(
i
=
0
;
i
<
max_size
;
i
++
)
for
(
i
=
0
;
i
<
max_size
;
i
++
)
nr_free
+=
(
s64
)(
32
-
hweight32
(
b
[
i
])
);
nr_free
-=
(
s64
)
hweight32
(
kaddr
[
i
]
);
kunmap
(
page
);
kunmap
_atomic
(
kaddr
,
KM_USER0
);
page_cache_release
(
page
);
page_cache_release
(
page
);
}
}
if
(
max_size
==
PAGE_CACHE_SIZE
>>
2
)
{
ntfs_debug
(
"Finished reading $Bitmap, last index = 0x%lx."
,
index
-
1
);
/*
/*
* Get the multiples of 4 bytes in use in the final partial
* Fixup for eventual bits outside logical ntfs volume (see function
* page
.
* description above)
.
*/
*/
max_size
=
((((
vol
->
nr_clusters
+
7
)
>>
3
)
&
~
PAGE_CACHE_MASK
)
if
(
vol
->
nr_clusters
&
63
)
+
3
)
>>
2
;
nr_free
+=
64
-
(
vol
->
nr_clusters
&
63
);
/* If there is a partial page go back and do it. */
if
(
max_size
)
{
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
);
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
);
nr_free
-=
PAGE_CACHE_SIZE
*
8
;
continue
;
}
}
if
(
index
==
max_index
)
{
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
))
ntfs_debug
(
"Finished reading $MFT/$BITMAP, last index = 0x%lx."
,
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"
,
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
;
}
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment