linux-ntfs-dev Mailing List for Linux NTFS file system support (Page 224)
Development moved to https://sourceforge.net/projects/ntfs-3g/
Brought to you by:
antona,
cha0smaster
You can subscribe to this list here.
2001 |
Jan
(1) |
Feb
(1) |
Mar
(23) |
Apr
(1) |
May
(16) |
Jun
(5) |
Jul
(23) |
Aug
(29) |
Sep
(16) |
Oct
(4) |
Nov
(12) |
Dec
(9) |
---|---|---|---|---|---|---|---|---|---|---|---|---|
2002 |
Jan
(8) |
Feb
(4) |
Mar
(20) |
Apr
(25) |
May
(19) |
Jun
(6) |
Jul
(43) |
Aug
(22) |
Sep
(7) |
Oct
(17) |
Nov
(26) |
Dec
(54) |
2003 |
Jan
(50) |
Feb
(51) |
Mar
(93) |
Apr
(105) |
May
(46) |
Jun
(62) |
Jul
(71) |
Aug
(28) |
Sep
(25) |
Oct
(28) |
Nov
(41) |
Dec
(17) |
2004 |
Jan
(39) |
Feb
(94) |
Mar
(139) |
Apr
(46) |
May
(103) |
Jun
(23) |
Jul
(74) |
Aug
(42) |
Sep
(127) |
Oct
(122) |
Nov
(80) |
Dec
(43) |
2005 |
Jan
(66) |
Feb
(35) |
Mar
(21) |
Apr
(42) |
May
(57) |
Jun
(136) |
Jul
(239) |
Aug
(255) |
Sep
(221) |
Oct
(295) |
Nov
(250) |
Dec
(66) |
2006 |
Jan
(116) |
Feb
(142) |
Mar
(66) |
Apr
(61) |
May
(56) |
Jun
(47) |
Jul
(82) |
Aug
(202) |
Sep
(106) |
Oct
(85) |
Nov
(130) |
Dec
(58) |
2007 |
Jan
(122) |
Feb
(7) |
Mar
(27) |
Apr
(19) |
May
(23) |
Jun
(17) |
Jul
(18) |
Aug
(19) |
Sep
(38) |
Oct
(39) |
Nov
(17) |
Dec
(6) |
2008 |
Jan
(8) |
Feb
(10) |
Mar
(6) |
Apr
(2) |
May
(29) |
Jun
(2) |
Jul
(16) |
Aug
(4) |
Sep
(7) |
Oct
(3) |
Nov
(6) |
Dec
(10) |
2009 |
Jan
(4) |
Feb
(4) |
Mar
(4) |
Apr
(12) |
May
(5) |
Jun
(8) |
Jul
(5) |
Aug
(17) |
Sep
(4) |
Oct
(7) |
Nov
(10) |
Dec
(18) |
2010 |
Jan
(7) |
Feb
(3) |
Mar
(6) |
Apr
(7) |
May
(3) |
Jun
(4) |
Jul
(2) |
Aug
(4) |
Sep
(12) |
Oct
|
Nov
(1) |
Dec
|
2011 |
Jan
(11) |
Feb
(5) |
Mar
(6) |
Apr
(9) |
May
|
Jun
(6) |
Jul
(14) |
Aug
(12) |
Sep
(1) |
Oct
(1) |
Nov
(1) |
Dec
(4) |
2012 |
Jan
(3) |
Feb
(5) |
Mar
(6) |
Apr
(9) |
May
(4) |
Jun
(8) |
Jul
(19) |
Aug
(12) |
Sep
(1) |
Oct
(3) |
Nov
(2) |
Dec
(2) |
2013 |
Jan
(2) |
Feb
(2) |
Mar
(3) |
Apr
(4) |
May
|
Jun
(2) |
Jul
|
Aug
|
Sep
|
Oct
(4) |
Nov
|
Dec
(4) |
2014 |
Jan
|
Feb
(2) |
Mar
(7) |
Apr
(2) |
May
(11) |
Jun
(27) |
Jul
(5) |
Aug
|
Sep
(1) |
Oct
(3) |
Nov
(9) |
Dec
|
2015 |
Jan
(3) |
Feb
|
Mar
(3) |
Apr
(2) |
May
(1) |
Jun
(1) |
Jul
(6) |
Aug
(1) |
Sep
(5) |
Oct
|
Nov
|
Dec
(5) |
2016 |
Jan
|
Feb
|
Mar
(3) |
Apr
(25) |
May
(4) |
Jun
|
Jul
|
Aug
|
Sep
|
Oct
|
Nov
(9) |
Dec
(32) |
2017 |
Jan
(44) |
Feb
(22) |
Mar
(12) |
Apr
|
May
(3) |
Jun
(9) |
Jul
(12) |
Aug
(3) |
Sep
|
Oct
(1) |
Nov
|
Dec
(5) |
2018 |
Jan
(5) |
Feb
(11) |
Mar
(3) |
Apr
(6) |
May
(3) |
Jun
(7) |
Jul
(5) |
Aug
|
Sep
|
Oct
(7) |
Nov
(2) |
Dec
|
2019 |
Jan
|
Feb
(1) |
Mar
|
Apr
(20) |
May
(1) |
Jun
(5) |
Jul
(10) |
Aug
(27) |
Sep
(2) |
Oct
|
Nov
(2) |
Dec
|
2020 |
Jan
(9) |
Feb
(8) |
Mar
(5) |
Apr
(20) |
May
(9) |
Jun
(8) |
Jul
(6) |
Aug
(6) |
Sep
(6) |
Oct
(151) |
Nov
(37) |
Dec
(59) |
2021 |
Jan
(46) |
Feb
(41) |
Mar
(55) |
Apr
(18) |
May
(33) |
Jun
(8) |
Jul
(28) |
Aug
(81) |
Sep
(9) |
Oct
(195) |
Nov
(48) |
Dec
(10) |
2022 |
Jan
(7) |
Feb
(1) |
Mar
(9) |
Apr
(8) |
May
(4) |
Jun
(102) |
Jul
(26) |
Aug
(52) |
Sep
(6) |
Oct
(27) |
Nov
(4) |
Dec
(22) |
2023 |
Jan
(9) |
Feb
(7) |
Mar
(19) |
Apr
(4) |
May
(30) |
Jun
(30) |
Jul
(27) |
Aug
(35) |
Sep
(78) |
Oct
(42) |
Nov
(20) |
Dec
(7) |
2024 |
Jan
(36) |
Feb
(8) |
Mar
(25) |
Apr
(27) |
May
(1) |
Jun
|
Jul
|
Aug
|
Sep
|
Oct
|
Nov
|
Dec
|
From: Anton A. <ai...@ca...> - 2004-11-10 13:46:03
|
This is patch 26/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/10 1.2026.1.66) NTFS: 2.1.22 - Many bug and race fixes and error handling improvements. - Cleanup fs/ntfs/aops.c::ntfs_{read,write}page() since we know that a resident attribute will be smaller than a page which makes the code simpler. Also make the code more tolerant to concurrent ->truncate. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/Documentation/filesystems/ntfs.txt b/Documentation/filesystems/ntfs.txt --- a/Documentation/filesystems/ntfs.txt 2004-11-10 13:45:56 +00:00 +++ b/Documentation/filesystems/ntfs.txt 2004-11-10 13:45:56 +00:00 @@ -432,6 +432,9 @@ Note, a technical ChangeLog aimed at kernel hackers is in fs/ntfs/ChangeLog. +2.1.22: + - Improve handling of ntfs volumes with errors. + - Fix various bugs and race conditions. 2.1.21: - Fix several race conditions and various other bugs. - Many internal cleanups, code reorganization, optimizations, and mft diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:56 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:56 +00:00 @@ -25,7 +25,7 @@ - Enable the code for setting the NT4 compatibility flag when we start making NTFS 1.2 specific modifications. -2.1.22-WIP +2.1.22 - Many bug and race fixes and error handling improvements. - Improve error handling in fs/ntfs/inode.c::ntfs_truncate(). - Change fs/ntfs/inode.c::ntfs_truncate() to return an error code @@ -85,6 +85,9 @@ complete runlist for the mft mirror is always mapped into memory. - Add creation of buffers to fs/ntfs/mft.c::ntfs_sync_mft_mirror(). - Improve error handling in fs/ntfs/aops.c::ntfs_{read,write}_block(). + - Cleanup fs/ntfs/aops.c::ntfs_{read,write}page() since we know that a + resident attribute will be smaller than a page which makes the code + simpler. Also make the code more tolerant to concurrent ->truncate. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/Makefile b/fs/ntfs/Makefile --- a/fs/ntfs/Makefile 2004-11-10 13:45:56 +00:00 +++ b/fs/ntfs/Makefile 2004-11-10 13:45:56 +00:00 @@ -6,7 +6,7 @@ index.o inode.o mft.o mst.o namei.o runlist.o super.o sysctl.o \ unistr.o upcase.o -EXTRA_CFLAGS = -DNTFS_VERSION=\"2.1.22-WIP\" +EXTRA_CFLAGS = -DNTFS_VERSION=\"2.1.22\" ifeq ($(CONFIG_NTFS_DEBUG),y) EXTRA_CFLAGS += -DDEBUG diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:45:56 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:45:56 +00:00 @@ -341,7 +341,7 @@ */ static int ntfs_readpage(struct file *file, struct page *page) { - s64 attr_pos; + loff_t i_size; ntfs_inode *ni, *base_ni; u8 *kaddr; ntfs_attr_search_ctx *ctx; @@ -350,7 +350,6 @@ int err = 0; BUG_ON(!PageLocked(page)); - /* * This can potentially happen because we clear PageUptodate() during * ntfs_writepage() of MstProtected() attributes. @@ -359,7 +358,6 @@ unlock_page(page); return 0; } - ni = NTFS_I(page->mapping->host); /* NInoNonResident() == NInoIndexAllocPresent() */ @@ -381,12 +379,23 @@ /* Normal data stream. */ return ntfs_read_block(page); } - /* Attribute is resident, implying it is not compressed or encrypted. */ + /* + * Attribute is resident, implying it is not compressed or encrypted. + * This also means the attribute is smaller than an mft record and + * hence smaller than a page, so can simply zero out any pages with + * index above 0. We can also do this if the file size is 0. + */ + if (unlikely(page->index > 0 || !i_size_read(VFS_I(ni)))) { + kaddr = kmap_atomic(page, KM_USER0); + memset(kaddr, 0, PAGE_CACHE_SIZE); + flush_dcache_page(page); + kunmap_atomic(kaddr, KM_USER0); + goto done; + } if (!NInoAttr(ni)) base_ni = ni; else base_ni = ni->ext.base_ntfs_ino; - /* Map, pin, and lock the mft record. */ mrec = map_mft_record(base_ni); if (IS_ERR(mrec)) { @@ -402,35 +411,25 @@ CASE_SENSITIVE, 0, NULL, 0, ctx); if (unlikely(err)) goto put_unm_err_out; - - /* Starting position of the page within the attribute value. */ - attr_pos = page->index << PAGE_CACHE_SHIFT; - - /* The total length of the attribute value. */ attr_len = le32_to_cpu(ctx->attr->data.resident.value_length); - + i_size = i_size_read(VFS_I(ni)); + if (unlikely(attr_len > i_size)) + attr_len = i_size; kaddr = kmap_atomic(page, KM_USER0); - /* Copy over in bounds data, zeroing the remainder of the page. */ - if (attr_pos < attr_len) { - u32 bytes = attr_len - attr_pos; - if (bytes > PAGE_CACHE_SIZE) - bytes = PAGE_CACHE_SIZE; - else if (bytes < PAGE_CACHE_SIZE) - memset(kaddr + bytes, 0, PAGE_CACHE_SIZE - bytes); - /* Copy the data to the page. */ - memcpy(kaddr, attr_pos + (char*)ctx->attr + - le16_to_cpu( - ctx->attr->data.resident.value_offset), bytes); - } else - memset(kaddr, 0, PAGE_CACHE_SIZE); + /* Copy the data to the page. */ + memcpy(kaddr, (u8*)ctx->attr + + le16_to_cpu(ctx->attr->data.resident.value_offset), + attr_len); + /* Zero the remainder of the page. */ + memset(kaddr + attr_len, 0, PAGE_CACHE_SIZE - attr_len); flush_dcache_page(page); kunmap_atomic(kaddr, KM_USER0); - - SetPageUptodate(page); put_unm_err_out: ntfs_attr_put_search_ctx(ctx); unm_err_out: unmap_mft_record(base_ni); +done: + SetPageUptodate(page); err_out: unlock_page(page); return err; @@ -1223,21 +1222,22 @@ */ static int ntfs_writepage(struct page *page, struct writeback_control *wbc) { - s64 attr_pos; + loff_t i_size; struct inode *vi; ntfs_inode *ni, *base_ni; char *kaddr; ntfs_attr_search_ctx *ctx; MFT_RECORD *m; - u32 attr_len, bytes; + u32 attr_len; int err; BUG_ON(!PageLocked(page)); vi = page->mapping->host; + i_size = i_size_read(vi); /* Is the page fully outside i_size? (truncate in progress) */ - if (unlikely(page->index >= (vi->i_size + PAGE_CACHE_SIZE - 1) >> + if (unlikely(page->index >= (i_size + PAGE_CACHE_SIZE - 1) >> PAGE_CACHE_SHIFT)) { /* * The page may have dirty, unmapped buffers. Make them @@ -1248,7 +1248,6 @@ ntfs_debug("Write outside i_size - truncated?"); return 0; } - ni = NTFS_I(vi); /* NInoNonResident() == NInoIndexAllocPresent() */ @@ -1284,9 +1283,9 @@ } } /* We have to zero every time due to mmap-at-end-of-file. */ - if (page->index >= (vi->i_size >> PAGE_CACHE_SHIFT)) { + if (page->index >= (i_size >> PAGE_CACHE_SHIFT)) { /* The page straddles i_size. */ - unsigned int ofs = vi->i_size & ~PAGE_CACHE_MASK; + unsigned int ofs = i_size & ~PAGE_CACHE_MASK; kaddr = kmap_atomic(page, KM_USER0); memset(kaddr + ofs, 0, PAGE_CACHE_SIZE - ofs); flush_dcache_page(page); @@ -1300,16 +1299,25 @@ } /* * Attribute is resident, implying it is not compressed, encrypted, - * sparse, or mst protected. + * sparse, or mst protected. This also means the attribute is smaller + * than an mft record and hence smaller than a page, so can simply + * return error on any pages with index above 0. */ BUG_ON(page_has_buffers(page)); BUG_ON(!PageUptodate(page)); - + if (unlikely(page->index > 0)) { + ntfs_error(vi->i_sb, "BUG()! page->index (0x%lx) > 0. " + "Aborting write.", page->index); + BUG_ON(PageWriteback(page)); + set_page_writeback(page); + unlock_page(page); + end_page_writeback(page); + return -EIO; + } if (!NInoAttr(ni)) base_ni = ni; else base_ni = ni->ext.base_ntfs_ino; - /* Map, pin, and lock the mft record. */ m = map_mft_record(base_ni); if (IS_ERR(m)) { @@ -1327,32 +1335,6 @@ CASE_SENSITIVE, 0, NULL, 0, ctx); if (unlikely(err)) goto err_out; - - /* Starting position of the page within the attribute value. */ - attr_pos = page->index << PAGE_CACHE_SHIFT; - - /* The total length of the attribute value. */ - attr_len = le32_to_cpu(ctx->attr->data.resident.value_length); - - if (unlikely(vi->i_size != attr_len)) { - ntfs_error(vi->i_sb, "BUG()! i_size (0x%llx) doesn't match " - "attr_len (0x%x). Aborting write.", vi->i_size, - attr_len); - err = -EIO; - goto err_out; - } - if (unlikely(attr_pos >= attr_len)) { - ntfs_error(vi->i_sb, "BUG()! attr_pos (0x%llx) > attr_len " - "(0x%x). Aborting write.", - (unsigned long long)attr_pos, attr_len); - err = -EIO; - goto err_out; - } - - bytes = attr_len - attr_pos; - if (unlikely(bytes > PAGE_CACHE_SIZE)) - bytes = PAGE_CACHE_SIZE; - /* * Keep the VM happy. This must be done otherwise the radix-tree tag * PAGECACHE_TAG_DIRTY remains set even though the page is clean. @@ -1384,26 +1366,30 @@ * TODO: ntfs_truncate(), others? */ + attr_len = le32_to_cpu(ctx->attr->data.resident.value_length); + i_size = i_size_read(VFS_I(ni)); kaddr = kmap_atomic(page, KM_USER0); + if (unlikely(attr_len > i_size)) { + /* Zero out of bounds area in the mft record. */ + memset((u8*)ctx->attr + le16_to_cpu( + ctx->attr->data.resident.value_offset) + + i_size, 0, attr_len - i_size); + attr_len = i_size; + } /* Copy the data from the page to the mft record. */ - memcpy((u8*)ctx->attr + le16_to_cpu( - ctx->attr->data.resident.value_offset) + attr_pos, - kaddr, bytes); + memcpy((u8*)ctx->attr + + le16_to_cpu(ctx->attr->data.resident.value_offset), + kaddr, attr_len); flush_dcache_mft_record_page(ctx->ntfs_ino); -#if 0 - /* Zero out of bounds area. */ - if (likely(bytes < PAGE_CACHE_SIZE)) { - memset(kaddr + bytes, 0, PAGE_CACHE_SIZE - bytes); - flush_dcache_page(page); - } -#endif + /* Zero out of bounds area in the page cache page. */ + memset(kaddr + attr_len, 0, PAGE_CACHE_SIZE - attr_len); + flush_dcache_page(page); kunmap_atomic(kaddr, KM_USER0); end_page_writeback(page); /* Mark the mft record dirty, so it gets written back. */ mark_mft_record_dirty(ctx->ntfs_ino); - ntfs_attr_put_search_ctx(ctx); unmap_mft_record(base_ni); return 0; @@ -1413,13 +1399,13 @@ "page so we try again later."); /* * Put the page back on mapping->dirty_pages, but leave its - * buffer's dirty state as-is. + * buffers' dirty state as-is. */ redirty_page_for_writepage(wbc, page); err = 0; } else { ntfs_error(vi->i_sb, "Resident attribute write failed with " - "error %i. Setting page error flag.", -err); + "error %i. Setting page error flag.", err); SetPageError(page); } unlock_page(page); |
From: Anton A. <ai...@ca...> - 2004-11-10 13:46:01
|
This is patch 24/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/09 1.2026.1.64) NTFS: - Fix creation of buffers in fs/ntfs/mft.c::ntfs_sync_mft_mirror(). Cannot just call fs/ntfs/aops.c::mark_ntfs_record_dirty() since this also marks the page dirty so we create the buffers by hand and set them uptodate. - Revert the removal of the page uptodate check from fs/ntfs/aops.c::mark_ntfs_record_dirty() as it is no longer called from fs/ntfs/mft.c::ntfs_sync_mft_mirror(). Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:48 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:48 +00:00 @@ -89,9 +89,6 @@ and write_mft_record_nolock(). From now on we require that the complete runlist for the mft mirror is always mapped into memory. - Add creation of buffers to fs/ntfs/mft.c::ntfs_sync_mft_mirror(). - - Do not check for the page being uptodate in mark_ntfs_record_dirty() - as we now call this after marking the page not uptodate during mft - mirror synchronisation (fs/ntfs/mft.c::ntfs_sync_mft_mirror()). - Improve error handling in fs/ntfs/aops.c::ntfs_{read,write}_block(). 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:45:48 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:45:48 +00:00 @@ -819,6 +819,12 @@ BUG_ON(!NInoNonResident(ni)); BUG_ON(!NInoMstProtected(ni)); is_mft = (S_ISREG(vi->i_mode) && !vi->i_ino); + /* + * NOTE: ntfs_write_mst_block() would be called for $MFTMirr if a page + * in its page cache were to be marked dirty. However this should + * never happen with the current driver and considering we do not + * handle this case here we do want to BUG(), at least for now. + */ BUG_ON(!(is_mft || S_ISDIR(vi->i_mode) || (NInoAttr(ni) && ni->type == AT_INDEX_ALLOCATION))); BUG_ON(!max_bhs); @@ -2282,6 +2288,7 @@ struct buffer_head *bh, *head, *buffers_to_free = NULL; unsigned int end, bh_size, bh_ofs; + BUG_ON(!PageUptodate(page)); end = ofs + ni->itype.index.block_size; bh_size = 1 << VFS_I(ni)->i_blkbits; spin_lock(&mapping->private_lock); diff -Nru a/fs/ntfs/mft.c b/fs/ntfs/mft.c --- a/fs/ntfs/mft.c 2004-11-10 13:45:48 +00:00 +++ b/fs/ntfs/mft.c 2004-11-10 13:45:48 +00:00 @@ -497,12 +497,20 @@ kmirr = page_address(page) + page_ofs; /* Copy the mst protected mft record to the mirror. */ memcpy(kmirr, m, vol->mft_record_size); - /* - * Create buffers if not present and mark the ones belonging to the mft - * mirror record dirty. - */ - mark_ntfs_record_dirty(page, page_ofs); - BUG_ON(!page_has_buffers(page)); + /* Create uptodate buffers if not present. */ + if (unlikely(!page_has_buffers(page))) { + struct buffer_head *tail; + + bh = head = alloc_page_buffers(page, blocksize, 1); + do { + set_buffer_uptodate(bh); + tail = bh; + bh = bh->b_this_page; + } while (bh); + tail->b_this_page = head; + attach_page_buffers(page, head); + BUG_ON(!page_has_buffers(page)); + } bh = head = page_buffers(page); BUG_ON(!bh); rl = NULL; |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:58
|
This is patch 25/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/09 1.2026.1.65) NTFS: Disable the file size changing code from fs/ntfs/aops.c::ntfs_prepare_write() for now as it is not safe. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:52 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:52 +00:00 @@ -49,11 +49,6 @@ - In fs/ntfs/aops.c::ntfs_writepage(), if the page is fully outside i_size, i.e. race with truncate, invalidate the buffers on the page so that they become freeable and hence the page does not leak. - - Implement extension of resident files in the regular file write code - paths (fs/ntfs/aops.c::ntfs_{prepare,commit}_write()). At present - this only works until the data attribute becomes too big for the mft - record after which we abort the write returning -EOPNOTSUPP from - ntfs_prepare_write(). - Remove unused function fs/ntfs/runlist.c::ntfs_rl_merge(). (Adrian Bunk) - Fix stupid bug in fs/ntfs/attrib.c::ntfs_attr_find() that resulted in diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:45:52 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:45:52 +00:00 @@ -1884,6 +1884,12 @@ /* If we do not need to resize the attribute allocation we are done. */ if (new_size <= vi->i_size) goto done; + + // FIXME: We abort for now as this code is not safe. + ntfs_error(vi->i_sb, "Changing the file size is not supported yet. " + "Sorry."); + return -EOPNOTSUPP; + /* Map, pin, and lock the (base) mft record. */ if (!NInoAttr(ni)) base_ni = ni; |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:53
|
This is patch 22/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/05 1.2026.1.57) NTFS: Minor cleanup of fs/ntfs/debug.c. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/debug.c b/fs/ntfs/debug.c --- a/fs/ntfs/debug.c 2004-11-10 13:45:40 +00:00 +++ b/fs/ntfs/debug.c 2004-11-10 13:45:40 +00:00 @@ -127,8 +127,8 @@ va_start(args, fmt); vsnprintf(err_buf, sizeof(err_buf), fmt, args); va_end(args); - printk(KERN_DEBUG "NTFS-fs DEBUG (%s, %d): %s(): %s\n", - file, line, flen ? function : "", err_buf); + printk(KERN_DEBUG "NTFS-fs DEBUG (%s, %d): %s(): %s\n", file, line, + flen ? function : "", err_buf); spin_unlock(&err_buf_lock); } @@ -141,8 +141,7 @@ if (!debug_msgs) return; - printk(KERN_DEBUG "NTFS-fs DEBUG: Dumping runlist (values " - "in hex):\n"); + printk(KERN_DEBUG "NTFS-fs DEBUG: Dumping runlist (values in hex):\n"); if (!rl) { printk(KERN_DEBUG "Run list not present.\n"); return; @@ -157,14 +156,14 @@ if (index > -LCN_ENOENT - 1) index = 3; printk(KERN_DEBUG "%-16Lx %s %-16Lx%s\n", - (rl + i)->vcn, lcn_str[index], - (rl + i)->length, (rl + i)->length ? - "" : " (runlist end)"); + (rl + i)->vcn, lcn_str[index], + (rl + i)->length, (rl + i)->length ? + "" : " (runlist end)"); } else printk(KERN_DEBUG "%-16Lx %-16Lx %-16Lx%s\n", - (rl + i)->vcn, (rl + i)->lcn, - (rl + i)->length, (rl + i)->length ? - "" : " (runlist end)"); + (rl + i)->vcn, (rl + i)->lcn, + (rl + i)->length, (rl + i)->length ? + "" : " (runlist end)"); if (!(rl + i)->length) break; } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:50
|
This is patch 23/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/09 1.2026.1.63) NTFS: - Add mapping of unmapped buffers to all remaining code paths, i.e. fs/ntfs/aops.c::ntfs_write_mst_block(), mft.c::ntfs_sync_mft_mirror(), and write_mft_record_nolock(). From now on we require that the complete runlist for the mft mirror is always mapped into memory. - Add creation of buffers to fs/ntfs/mft.c::ntfs_sync_mft_mirror(). - Do not check for the page being uptodate in mark_ntfs_record_dirty() as we now call this after marking the page not uptodate during mft mirror synchronisation (fs/ntfs/mft.c::ntfs_sync_mft_mirror()). - Improve error handling in fs/ntfs/aops.c::ntfs_{read,write}_block(). Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:44 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:44 +00:00 @@ -84,6 +84,15 @@ - Fix error handling in fs/ntfs/quota.c::ntfs_mark_quotas_out_of_date() where we failed to release i_sem on the $Quota/$Q attribute inode. - Fix bug in handling of bad inodes in fs/ntfs/namei.c::ntfs_lookup(). + - Add mapping of unmapped buffers to all remaining code paths, i.e. + fs/ntfs/aops.c::ntfs_write_mst_block(), mft.c::ntfs_sync_mft_mirror(), + and write_mft_record_nolock(). From now on we require that the + complete runlist for the mft mirror is always mapped into memory. + - Add creation of buffers to fs/ntfs/mft.c::ntfs_sync_mft_mirror(). + - Do not check for the page being uptodate in mark_ntfs_record_dirty() + as we now call this after marking the page not uptodate during mft + mirror synchronisation (fs/ntfs/mft.c::ntfs_sync_mft_mirror()). + - Improve error handling in fs/ntfs/aops.c::ntfs_{read,write}_block(). 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:45:44 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:45:44 +00:00 @@ -175,6 +175,9 @@ ni = NTFS_I(page->mapping->host); vol = ni->vol; + /* $MFT/$DATA must have its complete runlist in memory at all times. */ + BUG_ON(!ni->runlist.rl && !ni->mft_no && !NInoAttr(ni)); + blocksize_bits = VFS_I(ni)->i_blkbits; blocksize = 1 << blocksize_bits; @@ -190,12 +193,6 @@ lblock = (ni->allocated_size + blocksize - 1) >> blocksize_bits; zblock = (ni->initialized_size + blocksize - 1) >> blocksize_bits; -#ifdef DEBUG - if (unlikely(!ni->runlist.rl && !ni->mft_no && !NInoAttr(ni))) - panic("NTFS: $MFT/$DATA runlist has been unmapped! This is a " - "very serious bug! Cannot continue..."); -#endif - /* Loop through all the buffers in the page. */ rl = NULL; nr = i = 0; @@ -249,24 +246,30 @@ goto handle_hole; /* If first try and runlist unmapped, map and retry. */ if (!is_retry && lcn == LCN_RL_NOT_MAPPED) { + int err; is_retry = TRUE; /* * Attempt to map runlist, dropping lock for * the duration. */ up_read(&ni->runlist.lock); - if (!ntfs_map_runlist(ni, vcn)) + err = ntfs_map_runlist(ni, vcn); + if (likely(!err)) goto lock_retry_remap; rl = NULL; + lcn = err; } /* Hard error, zero out region. */ + bh->b_blocknr = -1; SetPageError(page); - ntfs_error(vol->sb, "ntfs_rl_vcn_to_lcn(vcn = 0x%llx) " - "failed with error code 0x%llx%s.", - (unsigned long long)vcn, - (unsigned long long)-lcn, - is_retry ? " even after retrying" : ""); - // FIXME: Depending on vol->on_errors, do something. + ntfs_error(vol->sb, "Failed to read from inode 0x%lx, " + "attribute type 0x%x, vcn 0x%llx, " + "offset 0x%x because its location on " + "disk could not be determined%s " + "(error code %lli).", ni->mft_no, + ni->type, (unsigned long long)vcn, + vcn_ofs, is_retry ? " even after " + "retrying" : "", (long long)lcn); } /* * Either iblock was outside lblock limits or @@ -437,8 +440,8 @@ /** * ntfs_write_block - write a @page to the backing store - * @wbc: writeback control structure * @page: page cache page to write out + * @wbc: writeback control structure * * This function is for writing pages belonging to non-resident, non-mst * protected attributes to their backing store. @@ -457,7 +460,7 @@ * * Based on ntfs_read_block() and __block_write_full_page(). */ -static int ntfs_write_block(struct writeback_control *wbc, struct page *page) +static int ntfs_write_block(struct page *page, struct writeback_control *wbc) { VCN vcn; LCN lcn; @@ -477,7 +480,7 @@ vol = ni->vol; ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index " - "0x%lx.", vi->i_ino, ni->type, page->index); + "0x%lx.", ni->mft_no, ni->type, page->index); BUG_ON(!NInoNonResident(ni)); BUG_ON(NInoMstProtected(ni)); @@ -618,9 +621,9 @@ bh->b_bdev = vol->sb->s_bdev; /* Convert block into corresponding vcn and offset. */ - vcn = (VCN)block << blocksize_bits >> vol->cluster_size_bits; - vcn_ofs = ((VCN)block << blocksize_bits) & - vol->cluster_size_mask; + vcn = (VCN)block << blocksize_bits; + vcn_ofs = vcn & vol->cluster_size_mask; + vcn >>= vol->cluster_size_bits; if (!rl) { lock_retry_remap: down_read(&ni->runlist.lock); @@ -663,15 +666,17 @@ if (likely(!err)) goto lock_retry_remap; rl = NULL; + lcn = err; } /* Failed to map the buffer, even after retrying. */ - bh->b_blocknr = -1UL; - ntfs_error(vol->sb, "ntfs_rl_vcn_to_lcn(vcn = 0x%llx) failed " - "with error code 0x%llx%s.", - (unsigned long long)vcn, - (unsigned long long)-lcn, - is_retry ? " even after retrying" : ""); - // FIXME: Depending on vol->on_errors, do something. + bh->b_blocknr = -1; + ntfs_error(vol->sb, "Failed to write to inode 0x%lx, " + "attribute type 0x%x, vcn 0x%llx, offset 0x%x " + "because its location on disk could not be " + "determined%s (error code %lli).", ni->mft_no, + ni->type, (unsigned long long)vcn, + vcn_ofs, is_retry ? " even after " + "retrying" : "", (long long)lcn); if (!err) err = -EIO; break; @@ -767,8 +772,8 @@ /** * ntfs_write_mst_block - write a @page to the backing store - * @wbc: writeback control structure * @page: page cache page to write out + * @wbc: writeback control structure * * This function is for writing pages belonging to non-resident, mst protected * attributes to their backing store. The only supported attributes are index @@ -789,22 +794,24 @@ * Based on ntfs_write_block(), ntfs_mft_writepage(), and * write_mft_record_nolock(). */ -static int ntfs_write_mst_block(struct writeback_control *wbc, - struct page *page) +static int ntfs_write_mst_block(struct page *page, + struct writeback_control *wbc) { sector_t block, dblock, rec_block; struct inode *vi = page->mapping->host; ntfs_inode *ni = NTFS_I(vi); ntfs_volume *vol = ni->vol; u8 *kaddr; - unsigned int bh_size = 1 << vi->i_blkbits; + unsigned char bh_size_bits = vi->i_blkbits; + unsigned int bh_size = 1 << bh_size_bits; unsigned int rec_size = ni->itype.index.block_size; ntfs_inode *locked_nis[PAGE_CACHE_SIZE / rec_size]; - struct buffer_head *bh, *head, *tbh; + struct buffer_head *bh, *head, *tbh, *rec_start_bh; int max_bhs = PAGE_CACHE_SIZE / bh_size; struct buffer_head *bhs[max_bhs]; - int i, nr_locked_nis, nr_recs, nr_bhs, bhs_per_rec, err; - unsigned char bh_size_bits, rec_size_bits; + runlist_element *rl; + int i, nr_locked_nis, nr_recs, nr_bhs, bhs_per_rec, err, err2; + unsigned rec_size_bits; BOOL sync, is_mft, page_is_dirty, rec_is_dirty; ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index " @@ -824,7 +831,6 @@ bh = head = page_buffers(page); BUG_ON(!bh); - bh_size_bits = vi->i_blkbits; rec_size_bits = ni->itype.index.block_size_bits; BUG_ON(!(PAGE_CACHE_SIZE >> rec_size_bits)); bhs_per_rec = rec_size >> bh_size_bits; @@ -837,25 +843,18 @@ /* The first out of bounds block for the data size. */ dblock = (vi->i_size + bh_size - 1) >> bh_size_bits; - err = nr_bhs = nr_recs = nr_locked_nis = 0; + rl = NULL; + err = err2 = nr_bhs = nr_recs = nr_locked_nis = 0; page_is_dirty = rec_is_dirty = FALSE; + rec_start_bh = NULL; do { - if (unlikely(block >= dblock)) { - /* - * Mapped buffers outside i_size will occur, because - * this page can be outside i_size when there is a - * truncate in progress. The contents of such buffers - * were zeroed by ntfs_writepage(). - * - * FIXME: What about the small race window where - * ntfs_writepage() has not done any clearing because - * the page was within i_size but before we get here, - * vmtruncate() modifies i_size? - */ - clear_buffer_dirty(bh); - continue; - } + BOOL is_retry = FALSE; + if (likely(block < rec_block)) { + if (unlikely(block >= dblock)) { + clear_buffer_dirty(bh); + continue; + } /* * This block is not the first one in the record. We * ignore the buffer's dirty state because we could @@ -863,22 +862,121 @@ */ if (!rec_is_dirty) continue; + if (unlikely(err2)) { + if (err2 != -ENOMEM) + clear_buffer_dirty(bh); + continue; + } } else /* if (block == rec_block) */ { BUG_ON(block > rec_block); /* This block is the first one in the record. */ rec_block += bhs_per_rec; + err2 = 0; + if (unlikely(block >= dblock)) { + clear_buffer_dirty(bh); + continue; + } if (!buffer_dirty(bh)) { /* Clean records are not written out. */ rec_is_dirty = FALSE; continue; } rec_is_dirty = TRUE; + rec_start_bh = bh; + } + /* Need to map the buffer if it is not mapped already. */ + if (unlikely(!buffer_mapped(bh))) { + VCN vcn; + LCN lcn; + unsigned int vcn_ofs; + + /* Obtain the vcn and offset of the current block. */ + vcn = (VCN)block << bh_size_bits; + vcn_ofs = vcn & vol->cluster_size_mask; + vcn >>= vol->cluster_size_bits; + if (!rl) { +lock_retry_remap: + down_read(&ni->runlist.lock); + rl = ni->runlist.rl; + } + if (likely(rl != NULL)) { + /* Seek to element containing target vcn. */ + while (rl->length && rl[1].vcn <= vcn) + rl++; + lcn = ntfs_rl_vcn_to_lcn(rl, vcn); + } else + lcn = LCN_RL_NOT_MAPPED; + /* Successful remap. */ + if (likely(lcn >= 0)) { + /* Setup buffer head to correct block. */ + bh->b_blocknr = ((lcn << + vol->cluster_size_bits) + + vcn_ofs) >> bh_size_bits; + set_buffer_mapped(bh); + } else { + /* + * Remap failed. Retry to map the runlist once + * unless we are working on $MFT which always + * has the whole of its runlist in memory. + */ + if (!is_mft && !is_retry && + lcn == LCN_RL_NOT_MAPPED) { + is_retry = TRUE; + /* + * Attempt to map runlist, dropping + * lock for the duration. + */ + up_read(&ni->runlist.lock); + err2 = ntfs_map_runlist(ni, vcn); + if (likely(!err2)) + goto lock_retry_remap; + if (err2 == -ENOMEM) + page_is_dirty = TRUE; + lcn = err2; + } else + err2 = -EIO; + /* Hard error. Abort writing this record. */ + if (!err || err == -ENOMEM) + err = err2; + bh->b_blocknr = -1; + ntfs_error(vol->sb, "Cannot write ntfs record " + "0x%llx (inode 0x%lx, " + "attribute type 0x%x) because " + "its location on disk could " + "not be determined (error " + "code %lli).", (s64)block << + bh_size_bits >> + vol->mft_record_size_bits, + ni->mft_no, ni->type, + (long long)lcn); + /* + * If this is not the first buffer, remove the + * buffers in this record from the list of + * buffers to write and clear their dirty bit + * if not error -ENOMEM. + */ + if (rec_start_bh != bh) { + while (bhs[--nr_bhs] != rec_start_bh) + ; + if (err2 != -ENOMEM) { + do { + clear_buffer_dirty( + rec_start_bh); + } while ((rec_start_bh = + rec_start_bh-> + b_this_page) != + bh); + } + } + continue; + } } - BUG_ON(!buffer_mapped(bh)); BUG_ON(!buffer_uptodate(bh)); + BUG_ON(nr_bhs >= max_bhs); bhs[nr_bhs++] = bh; - BUG_ON(nr_bhs > max_bhs); } while (block++, (bh = bh->b_this_page) != head); + if (unlikely(rl)) + up_read(&ni->runlist.lock); /* If there were no dirty buffers, we are done. */ if (!nr_bhs) goto done; @@ -930,9 +1028,11 @@ locked_nis[nr_locked_nis++] = tni; } /* Apply the mst protection fixups. */ - err = pre_write_mst_fixup((NTFS_RECORD*)(kaddr + ofs), + err2 = pre_write_mst_fixup((NTFS_RECORD*)(kaddr + ofs), rec_size); - if (unlikely(err)) { + if (unlikely(err2)) { + if (!err || err == -ENOMEM) + err = -EIO; ntfs_error(vol->sb, "Failed to apply mst fixups " "(inode 0x%lx, attribute type 0x%x, " "page index 0x%lx, page offset 0x%x)!" @@ -986,7 +1086,8 @@ "0x%lx, page offset 0x%lx)! Unmount " "and run chkdsk.", vi->i_ino, ni->type, page->index, bh_offset(tbh)); - err = -EIO; + if (!err || err == -ENOMEM) + err = -EIO; /* * Set the buffer uptodate so the page and buffer * states do not become out of sync. @@ -1056,13 +1157,18 @@ atomic_dec(&tni->count); iput(VFS_I(base_tni)); } - if (unlikely(err)) { - SetPageError(page); - NVolSetErrors(vol); - } SetPageUptodate(page); kunmap(page); done: + if (unlikely(err && err != -ENOMEM)) { + /* + * Set page error if there is only one ntfs record in the page. + * Otherwise we would loose per-record granularity. + */ + if (ni->itype.index.block_size == PAGE_CACHE_SIZE) + SetPageError(page); + NVolSetErrors(vol); + } if (page_is_dirty) { ntfs_debug("Page still contains one or more dirty ntfs " "records. Redirtying the page starting at " @@ -1182,9 +1288,9 @@ } /* Handle mst protected attributes. */ if (NInoMstProtected(ni)) - return ntfs_write_mst_block(wbc, page); + return ntfs_write_mst_block(page, wbc); /* Normal data stream. */ - return ntfs_write_block(wbc, page); + return ntfs_write_block(page, wbc); } /* * Attribute is resident, implying it is not compressed, encrypted, @@ -1343,7 +1449,7 @@ vol = ni->vol; ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index " - "0x%lx, from = %u, to = %u.", vi->i_ino, ni->type, + "0x%lx, from = %u, to = %u.", ni->mft_no, ni->type, page->index, from, to); BUG_ON(!NInoNonResident(ni)); @@ -1537,21 +1643,24 @@ if (likely(!err)) goto lock_retry_remap; rl = NULL; + lcn = err; } /* * Failed to map the buffer, even after * retrying. */ - bh->b_blocknr = -1UL; - ntfs_error(vol->sb, "ntfs_rl_vcn_to_lcn(vcn = " - "0x%llx) failed with error " - "code 0x%llx%s.", + bh->b_blocknr = -1; + ntfs_error(vol->sb, "Failed to write to inode " + "0x%lx, attribute type 0x%x, " + "vcn 0x%llx, offset 0x%x " + "because its location on disk " + "could not be determined%s " + "(error code %lli).", + ni->mft_no, ni->type, (unsigned long long)vcn, - (unsigned long long)-lcn, - is_retry ? " even after " - "retrying" : ""); - // FIXME: Depending on vol->on_errors, do - // something. + vcn_ofs, is_retry ? " even " + "after retrying" : "", + (long long)lcn); if (!err) err = -EIO; goto err_out; @@ -2173,7 +2282,6 @@ struct buffer_head *bh, *head, *buffers_to_free = NULL; unsigned int end, bh_size, bh_ofs; - BUG_ON(!PageUptodate(page)); end = ofs + ni->itype.index.block_size; bh_size = 1 << VFS_I(ni)->i_blkbits; spin_lock(&mapping->private_lock); diff -Nru a/fs/ntfs/inode.c b/fs/ntfs/inode.c --- a/fs/ntfs/inode.c 2004-11-10 13:45:44 +00:00 +++ b/fs/ntfs/inode.c 2004-11-10 13:45:44 +00:00 @@ -2358,8 +2358,8 @@ done: ntfs_attr_put_search_ctx(ctx); unmap_mft_record(ni); - ntfs_debug("Done."); NInoClearTruncateFailed(ni); + ntfs_debug("Done."); return 0; err_out: if (err != -ENOMEM) { @@ -2608,6 +2608,7 @@ ntfs_error(vi->i_sb, "Failed (error code %i): Marking inode " "as bad. You should run chkdsk.", -err); make_bad_inode(vi); + NVolSetErrors(ni->vol); } return err; } diff -Nru a/fs/ntfs/mft.c b/fs/ntfs/mft.c --- a/fs/ntfs/mft.c 2004-11-10 13:45:44 +00:00 +++ b/fs/ntfs/mft.c 2004-11-10 13:45:44 +00:00 @@ -466,8 +466,10 @@ struct buffer_head *bhs[max_bhs]; struct buffer_head *bh, *head; u8 *kmirr; - unsigned int block_start, block_end, m_start, m_end; + runlist_element *rl; + unsigned int block_start, block_end, m_start, m_end, page_ofs; int i_bhs, nr_bhs, err = 0; + unsigned char blocksize_bits = vol->mftmirr_ino->i_blkbits; ntfs_debug("Entering for inode 0x%lx.", mft_no); BUG_ON(!max_bhs); @@ -486,24 +488,24 @@ err = PTR_ERR(page); goto err_out; } - /* - * Exclusion against other writers. This should never be a problem - * since the page in which the mft record @m resides is also locked and - * hence any other writers would be held up there but it is better to - * make sure no one is writing from elsewhere. - */ lock_page(page); BUG_ON(!PageUptodate(page)); ClearPageUptodate(page); + /* Offset of the mft mirror record inside the page. */ + page_ofs = (mft_no << vol->mft_record_size_bits) & ~PAGE_CACHE_MASK; /* The address in the page of the mirror copy of the mft record @m. */ - kmirr = page_address(page) + ((mft_no << vol->mft_record_size_bits) & - ~PAGE_CACHE_MASK); + kmirr = page_address(page) + page_ofs; /* Copy the mst protected mft record to the mirror. */ memcpy(kmirr, m, vol->mft_record_size); - /* Make sure we have mapped buffers. */ + /* + * Create buffers if not present and mark the ones belonging to the mft + * mirror record dirty. + */ + mark_ntfs_record_dirty(page, page_ofs); BUG_ON(!page_has_buffers(page)); bh = head = page_buffers(page); BUG_ON(!bh); + rl = NULL; nr_bhs = 0; block_start = 0; m_start = kmirr - (u8*)page_address(page); @@ -511,15 +513,61 @@ do { block_end = block_start + blocksize; /* If the buffer is outside the mft record, skip it. */ - if ((block_end <= m_start) || (block_start >= m_end)) + if (block_end <= m_start) continue; - BUG_ON(!buffer_mapped(bh)); + if (unlikely(block_start >= m_end)) + break; + /* Need to map the buffer if it is not mapped already. */ + if (unlikely(!buffer_mapped(bh))) { + VCN vcn; + LCN lcn; + unsigned int vcn_ofs; + + /* Obtain the vcn and offset of the current block. */ + vcn = ((VCN)mft_no << vol->mft_record_size_bits) + + (block_start - m_start); + vcn_ofs = vcn & vol->cluster_size_mask; + vcn >>= vol->cluster_size_bits; + if (!rl) { + down_read(&NTFS_I(vol->mftmirr_ino)-> + runlist.lock); + rl = NTFS_I(vol->mftmirr_ino)->runlist.rl; + /* + * $MFTMirr always has the whole of its runlist + * in memory. + */ + BUG_ON(!rl); + } + /* Seek to element containing target vcn. */ + while (rl->length && rl[1].vcn <= vcn) + rl++; + lcn = ntfs_rl_vcn_to_lcn(rl, vcn); + /* For $MFTMirr, only lcn >= 0 is a successful remap. */ + if (likely(lcn >= 0)) { + /* Setup buffer head to correct block. */ + bh->b_blocknr = ((lcn << + vol->cluster_size_bits) + + vcn_ofs) >> blocksize_bits; + set_buffer_mapped(bh); + } else { + bh->b_blocknr = -1; + ntfs_error(vol->sb, "Cannot write mft mirror " + "record 0x%lx because its " + "location on disk could not " + "be determined (error code " + "%lli).", mft_no, + (long long)lcn); + err = -EIO; + } + } BUG_ON(!buffer_uptodate(bh)); BUG_ON(!nr_bhs && (m_start != block_start)); BUG_ON(nr_bhs >= max_bhs); bhs[nr_bhs++] = bh; BUG_ON((nr_bhs >= max_bhs) && (m_end != block_end)); } while (block_start = block_end, (bh = bh->b_this_page) != head); + if (unlikely(rl)) + up_read(&NTFS_I(vol->mftmirr_ino)->runlist.lock); if (likely(!err)) { /* Lock buffers and start synchronous write i/o on them. */ for (i_bhs = 0; i_bhs < nr_bhs; i_bhs++) { @@ -528,8 +576,7 @@ if (unlikely(test_set_buffer_locked(tbh))) BUG(); BUG_ON(!buffer_uptodate(tbh)); - if (buffer_dirty(tbh)) - clear_buffer_dirty(tbh); + clear_buffer_dirty(tbh); get_bh(tbh); tbh->b_end_io = end_buffer_write_sync; submit_bh(WRITE, tbh); @@ -613,13 +660,14 @@ { ntfs_volume *vol = ni->vol; struct page *page = ni->page; - unsigned int blocksize = vol->sb->s_blocksize; + unsigned char blocksize_bits = vol->mft_ino->i_blkbits; + unsigned int blocksize = 1 << blocksize_bits; int max_bhs = vol->mft_record_size / blocksize; struct buffer_head *bhs[max_bhs]; struct buffer_head *bh, *head; + runlist_element *rl; unsigned int block_start, block_end, m_start, m_end; int i_bhs, nr_bhs, err = 0; - BOOL rec_is_dirty = TRUE; ntfs_debug("Entering for inode 0x%lx.", ni->mft_no); BUG_ON(NInoAttr(ni)); @@ -636,6 +684,7 @@ BUG_ON(!page_has_buffers(page)); bh = head = page_buffers(page); BUG_ON(!bh); + rl = NULL; nr_bhs = 0; block_start = 0; m_start = ni->page_ofs; @@ -647,31 +696,65 @@ continue; if (unlikely(block_start >= m_end)) break; + /* + * If this block is not the first one in the record, we ignore + * the buffer's dirty state because we could have raced with a + * parallel mark_ntfs_record_dirty(). + */ if (block_start == m_start) { /* This block is the first one in the record. */ if (!buffer_dirty(bh)) { + BUG_ON(nr_bhs); /* Clean records are not written out. */ - rec_is_dirty = FALSE; - continue; + break; + } + } + /* Need to map the buffer if it is not mapped already. */ + if (unlikely(!buffer_mapped(bh))) { + VCN vcn; + LCN lcn; + unsigned int vcn_ofs; + + /* Obtain the vcn and offset of the current block. */ + vcn = ((VCN)ni->mft_no << vol->mft_record_size_bits) + + (block_start - m_start); + vcn_ofs = vcn & vol->cluster_size_mask; + vcn >>= vol->cluster_size_bits; + if (!rl) { + down_read(&NTFS_I(vol->mft_ino)->runlist.lock); + rl = NTFS_I(vol->mft_ino)->runlist.rl; + BUG_ON(!rl); + } + /* Seek to element containing target vcn. */ + while (rl->length && rl[1].vcn <= vcn) + rl++; + lcn = ntfs_rl_vcn_to_lcn(rl, vcn); + /* For $MFT, only lcn >= 0 is a successful remap. */ + if (likely(lcn >= 0)) { + /* Setup buffer head to correct block. */ + bh->b_blocknr = ((lcn << + vol->cluster_size_bits) + + vcn_ofs) >> blocksize_bits; + set_buffer_mapped(bh); + } else { + bh->b_blocknr = -1; + ntfs_error(vol->sb, "Cannot write mft record " + "0x%lx because its location " + "on disk could not be " + "determined (error code %lli).", + ni->mft_no, (long long)lcn); + err = -EIO; } - rec_is_dirty = TRUE; - } else { - /* - * This block is not the first one in the record. We - * ignore the buffer's dirty state because we could - * have raced with a parallel mark_ntfs_record_dirty(). - */ - if (!rec_is_dirty) - continue; } - BUG_ON(!buffer_mapped(bh)); BUG_ON(!buffer_uptodate(bh)); BUG_ON(!nr_bhs && (m_start != block_start)); BUG_ON(nr_bhs >= max_bhs); bhs[nr_bhs++] = bh; BUG_ON((nr_bhs >= max_bhs) && (m_end != block_end)); } while (block_start = block_end, (bh = bh->b_this_page) != head); - if (!rec_is_dirty) + if (unlikely(rl)) + up_read(&NTFS_I(vol->mft_ino)->runlist.lock); + if (!nr_bhs) goto done; if (unlikely(err)) goto cleanup_out; @@ -745,7 +828,8 @@ "Redirtying so the write is retried later."); mark_mft_record_dirty(ni); err = 0; - } + } else + NVolSetErrors(vol); return err; } diff -Nru a/fs/ntfs/super.c b/fs/ntfs/super.c --- a/fs/ntfs/super.c 2004-11-10 13:45:44 +00:00 +++ b/fs/ntfs/super.c 2004-11-10 13:45:44 +00:00 @@ -983,6 +983,10 @@ * @vol: ntfs super block describing device whose mft mirror to check * * Return TRUE on success or FALSE on error. + * + * Note, this function also results in the mft mirror runlist being completely + * mapped into memory. The mft mirror write code requires this and will BUG() + * should it find an unmapped runlist element. */ static BOOL check_mft_mirror(ntfs_volume *vol) { |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:41
|
This is patch 21/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/05 1.2026.1.56) NTFS: Fix bug in handling of bad inodes in fs/ntfs/namei.c::ntfs_lookup(). Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:36 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:36 +00:00 @@ -83,6 +83,7 @@ the bug report.) - Fix error handling in fs/ntfs/quota.c::ntfs_mark_quotas_out_of_date() where we failed to release i_sem on the $Quota/$Q attribute inode. + - Fix bug in handling of bad inodes in fs/ntfs/namei.c::ntfs_lookup(). 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/namei.c b/fs/ntfs/namei.c --- a/fs/ntfs/namei.c 2004-11-10 13:45:36 +00:00 +++ b/fs/ntfs/namei.c 2004-11-10 13:45:36 +00:00 @@ -127,15 +127,16 @@ dent_inode = ntfs_iget(vol->sb, dent_ino); if (likely(!IS_ERR(dent_inode))) { /* Consistency check. */ - if (MSEQNO(mref) == NTFS_I(dent_inode)->seq_no || + if (is_bad_inode(dent_inode) || MSEQNO(mref) == + NTFS_I(dent_inode)->seq_no || dent_ino == FILE_MFT) { /* Perfect WIN32/POSIX match. -- Case 1. */ if (!name) { - ntfs_debug("Done."); + ntfs_debug("Done. (Case 1.)"); return d_splice_alias(dent_inode, dent); } /* - * We are too indented. Handle imperfect + * We are too indented. Handle imperfect * matches and short file names further below. */ goto handle_name; @@ -181,6 +182,7 @@ nls_name.name = NULL; if (name->type != FILE_NAME_DOS) { /* Case 2. */ + ntfs_debug("Case 2."); nls_name.len = (unsigned)ntfs_ucstonls(vol, (ntfschar*)&name->name, name->len, (unsigned char**)&nls_name.name, 0); @@ -188,6 +190,7 @@ } else /* if (name->type == FILE_NAME_DOS) */ { /* Case 3. */ FILE_NAME_ATTR *fn; + ntfs_debug("Case 3."); kfree(name); /* Find the WIN32 name corresponding to the matched DOS name. */ @@ -271,12 +274,17 @@ dput(real_dent); else new_dent = real_dent; + ntfs_debug("Done. (Created new dentry.)"); return new_dent; } kfree(nls_name.name); /* Matching dentry exists, check if it is negative. */ if (real_dent->d_inode) { - BUG_ON(real_dent->d_inode != dent_inode); + if (unlikely(real_dent->d_inode != dent_inode)) { + /* This can happen because bad inodes are unhashed. */ + BUG_ON(!is_bad_inode(dent_inode)); + BUG_ON(!is_bad_inode(real_dent->d_inode)); + } /* * Already have the inode and the dentry attached, decrement * the reference count to balance the ntfs_iget() we did @@ -285,6 +293,7 @@ * about any NFS/disconnectedness issues here. */ iput(dent_inode); + ntfs_debug("Done. (Already had inode and dentry.)"); return real_dent; } /* @@ -295,6 +304,7 @@ if (!S_ISDIR(dent_inode->i_mode)) { /* Not a directory; everything is easy. */ d_instantiate(real_dent, dent_inode); + ntfs_debug("Done. (Already had negative file dentry.)"); return real_dent; } spin_lock(&dcache_lock); @@ -308,6 +318,7 @@ real_dent->d_inode = dent_inode; spin_unlock(&dcache_lock); security_d_instantiate(real_dent, dent_inode); + ntfs_debug("Done. (Already had negative directory dentry.)"); return real_dent; } /* @@ -327,6 +338,8 @@ /* Throw away real_dent. */ dput(real_dent); /* Use new_dent as the actual dentry. */ + ntfs_debug("Done. (Already had negative, disconnected directory " + "dentry.)"); return new_dent; eio_err_out: @@ -338,6 +351,7 @@ if (m) unmap_mft_record(ni); iput(dent_inode); + ntfs_error(vol->sb, "Failed, returning error code %i.", err); return ERR_PTR(err); } } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:40
|
This is patch 20/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/05 1.2026.1.55) NTFS: Fix error handling in fs/ntfs/quota.c::ntfs_mark_quotas_out_of_date() where we failed to release i_sem on the $Quota/$Q attribute inode. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:32 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:32 +00:00 @@ -81,6 +81,8 @@ where we cannot access any of the ntfs records in a page when a single one of them had an mst error. (Thanks to Ken MacFerrin for the bug report.) + - Fix error handling in fs/ntfs/quota.c::ntfs_mark_quotas_out_of_date() + where we failed to release i_sem on the $Quota/$Q attribute inode. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/quota.c b/fs/ntfs/quota.c --- a/fs/ntfs/quota.c 2004-11-10 13:45:32 +00:00 +++ b/fs/ntfs/quota.c 2004-11-10 13:45:32 +00:00 @@ -52,7 +52,7 @@ ictx = ntfs_index_ctx_get(NTFS_I(vol->quota_q_ino)); if (!ictx) { ntfs_error(vol->sb, "Failed to get index context."); - return FALSE; + goto err_out; } err = ntfs_index_lookup(&qid, sizeof(qid), ictx); if (err) { @@ -108,7 +108,8 @@ ntfs_debug("Done."); return TRUE; err_out: - ntfs_index_ctx_put(ictx); + if (ictx) + ntfs_index_ctx_put(ictx); up(&vol->quota_q_ino->i_sem); return FALSE; } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:37
|
This is patch 14/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/01 1.2026.1.44) NTFS: Add MODULE_VERSION() to fs/ntfs/super.c. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/super.c b/fs/ntfs/super.c --- a/fs/ntfs/super.c 2004-11-10 13:45:08 +00:00 +++ b/fs/ntfs/super.c 2004-11-10 13:45:08 +00:00 @@ -2754,6 +2754,7 @@ MODULE_AUTHOR("Anton Altaparmakov <ai...@ca...>"); MODULE_DESCRIPTION("NTFS 1.2/3.x driver - Copyright (c) 2001-2004 Anton Altaparmakov"); +MODULE_VERSION(NTFS_VERSION); MODULE_LICENSE("GPL"); #ifdef DEBUG module_param(debug_msgs, bool, 0); |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:34
|
This is patch 19/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/05 1.2026.1.54) NTFS: Rewrite handling of multi sector transfer errors. We now do not set PageError() when such errors are detected in the async i/o handler fs/ntfs/aops.c::ntfs_end_buffer_async_read(). All users of mst protected attributes now check the magic of each ntfs record as they use it and act appropriately. This has the effect of making errors granular per ntfs record rather than per page which solves the case where we cannot access any of the ntfs records in a page when a single one of them had an mst error. (Thanks to Ken MacFerrin for the bug report.) Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:28 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:28 +00:00 @@ -72,6 +72,15 @@ thus was not mapped. (Thanks to Ken MacFerrin for the bug report.) - Drop the runlist lock after the vcn has been read in fs/ntfs/lcnalloc.c::__ntfs_cluster_free(). + - Rewrite handling of multi sector transfer errors. We now do not set + PageError() when such errors are detected in the async i/o handler + fs/ntfs/aops.c::ntfs_end_buffer_async_read(). All users of mst + protected attributes now check the magic of each ntfs record as they + use it and act appropriately. This has the effect of making errors + granular per ntfs record rather than per page which solves the case + where we cannot access any of the ntfs records in a page when a + single one of them had an mst error. (Thanks to Ken MacFerrin for + the bug report.) 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:45:28 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:45:28 +00:00 @@ -43,13 +43,13 @@ * @uptodate: whether @bh is now uptodate or not * * Asynchronous I/O completion handler for reading pages belonging to the - * attribute address space of an inode. The inodes can either be files or + * attribute address space of an inode. The inodes can either be files or * directories or they can be fake inodes describing some attribute. * * If NInoMstProtected(), perform the post read mst fixups when all IO on the * page has been completed and mark the page uptodate or set the error bit on - * the page. To determine the size of the records that need fixing up, we cheat - * a little bit by setting the index_block_size in ntfs_inode to the ntfs + * the page. To determine the size of the records that need fixing up, we + * cheat a little bit by setting the index_block_size in ntfs_inode to the ntfs * record size, and index_block_size_bits, to the log(base 2) of the ntfs * record size. */ @@ -90,7 +90,6 @@ (unsigned long long)bh->b_blocknr); SetPageError(page); } - spin_lock_irqsave(&page_uptodate_lock, flags); clear_buffer_async_read(bh); unlock_buffer(bh); @@ -111,42 +110,30 @@ * If none of the buffers had errors then we can set the page uptodate, * but we first have to perform the post read mst fixups, if the * attribute is mst protected, i.e. if NInoMstProteced(ni) is true. + * Note we ignore fixup errors as those are detected when + * map_mft_record() is called which gives us per record granularity + * rather than per page granularity. */ if (!NInoMstProtected(ni)) { if (likely(page_uptodate && !PageError(page))) SetPageUptodate(page); } else { char *addr; - unsigned int i, recs, nr_err; + unsigned int i, recs; u32 rec_size; rec_size = ni->itype.index.block_size; recs = PAGE_CACHE_SIZE / rec_size; + /* Should have been verified before we got here... */ + BUG_ON(!recs); addr = kmap_atomic(page, KM_BIO_SRC_IRQ); - for (i = nr_err = 0; i < recs; i++) { - if (likely(!post_read_mst_fixup((NTFS_RECORD*)(addr + - i * rec_size), rec_size))) - continue; - nr_err++; - ntfs_error(ni->vol->sb, "post_read_mst_fixup() failed, " - "corrupt %s record 0x%llx. Run chkdsk.", - ni->mft_no ? "index" : "mft", - (unsigned long long)(((s64)page->index - << PAGE_CACHE_SHIFT >> - ni->itype.index.block_size_bits) + i)); - } + for (i = 0; i < recs; i++) + post_read_mst_fixup((NTFS_RECORD*)(addr + + i * rec_size), rec_size); flush_dcache_page(page); kunmap_atomic(addr, KM_BIO_SRC_IRQ); - if (likely(!PageError(page))) { - if (likely(!nr_err && recs)) { - if (likely(page_uptodate)) - SetPageUptodate(page); - } else { - ntfs_error(ni->vol->sb, "Setting page error, " - "index 0x%lx.", page->index); - SetPageError(page); - } - } + if (likely(!PageError(page) && page_uptodate)) + SetPageUptodate(page); } unlock_page(page); return; diff -Nru a/fs/ntfs/aops.h b/fs/ntfs/aops.h --- a/fs/ntfs/aops.h 2004-11-10 13:45:28 +00:00 +++ b/fs/ntfs/aops.h 2004-11-10 13:45:28 +00:00 @@ -55,6 +55,13 @@ * method defined in the address space operations of @mapping and the page is * added to the page cache of @mapping in the process. * + * If the page belongs to an mst protected attribute and it is marked as such + * in its ntfs inode (NInoMstProtected()) the mst fixups are applied but no + * error checking is performed. This means the caller has to verify whether + * the ntfs record(s) contained in the page are valid or not using one of the + * ntfs_is_XXXX_record{,p}() macros, where XXXX is the record type you are + * expecting to see. (For details of the macros, see fs/ntfs/layout.h.) + * * If the page is in high memory it is mapped into memory directly addressible * by the kernel. * diff -Nru a/fs/ntfs/dir.c b/fs/ntfs/dir.c --- a/fs/ntfs/dir.c 2004-11-10 13:45:28 +00:00 +++ b/fs/ntfs/dir.c 2004-11-10 13:45:28 +00:00 @@ -300,7 +300,6 @@ ntfs_error(sb, "No index allocation attribute but index entry " "requires one. Directory inode 0x%lx is " "corrupt or driver bug.", dir_ni->mft_no); - err = -EIO; goto err_out; } /* Get the starting vcn of the index_block holding the child node. */ @@ -338,7 +337,13 @@ if ((u8*)ia < kaddr || (u8*)ia > kaddr + PAGE_CACHE_SIZE) { ntfs_error(sb, "Out of bounds check failed. Corrupt directory " "inode 0x%lx or driver bug.", dir_ni->mft_no); - err = -EIO; + goto unm_err_out; + } + /* Catch multi sector transfer fixup errors. */ + if (unlikely(!ntfs_is_indx_record(ia->magic))) { + ntfs_error(sb, "Directory index record with vcn 0x%llx is " + "corrupt. Corrupt inode 0x%lx. Run chkdsk.", + (unsigned long long)vcn, dir_ni->mft_no); goto unm_err_out; } if (sle64_to_cpu(ia->index_block_vcn) != vcn) { @@ -348,7 +353,6 @@ "bug.", (unsigned long long) sle64_to_cpu(ia->index_block_vcn), (unsigned long long)vcn, dir_ni->mft_no); - err = -EIO; goto unm_err_out; } if (le32_to_cpu(ia->index.allocated_size) + 0x18 != @@ -360,7 +364,6 @@ (unsigned long long)vcn, dir_ni->mft_no, le32_to_cpu(ia->index.allocated_size) + 0x18, dir_ni->itype.index.block_size); - err = -EIO; goto unm_err_out; } index_end = (u8*)ia + dir_ni->itype.index.block_size; @@ -370,7 +373,6 @@ "Cannot access! This is probably a bug in the " "driver.", (unsigned long long)vcn, dir_ni->mft_no); - err = -EIO; goto unm_err_out; } index_end = (u8*)&ia->index + le32_to_cpu(ia->index.index_length); @@ -378,7 +380,6 @@ ntfs_error(sb, "Size of index buffer (VCN 0x%llx) of directory " "inode 0x%lx exceeds maximum size.", (unsigned long long)vcn, dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* The first index entry. */ @@ -398,7 +399,6 @@ ntfs_error(sb, "Index entry out of bounds in " "directory inode 0x%lx.", dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* @@ -551,7 +551,6 @@ ntfs_error(sb, "Index entry with child node found in " "a leaf node in directory inode 0x%lx.", dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* Child node present, descend into it. */ @@ -572,7 +571,6 @@ } ntfs_error(sb, "Negative child node vcn in directory inode " "0x%lx.", dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* @@ -591,6 +589,8 @@ unlock_page(page); ntfs_unmap_page(page); err_out: + if (!err) + err = -EIO; if (ctx) ntfs_attr_put_search_ctx(ctx); if (m) @@ -601,8 +601,7 @@ } return ERR_MREF(err); dir_err_out: - ntfs_error(sb, "Corrupt directory. Aborting lookup."); - err = -EIO; + ntfs_error(sb, "Corrupt directory. Aborting lookup."); goto err_out; } @@ -780,7 +779,6 @@ ntfs_error(sb, "No index allocation attribute but index entry " "requires one. Directory inode 0x%lx is " "corrupt or driver bug.", dir_ni->mft_no); - err = -EIO; goto err_out; } /* Get the starting vcn of the index_block holding the child node. */ @@ -818,7 +816,13 @@ if ((u8*)ia < kaddr || (u8*)ia > kaddr + PAGE_CACHE_SIZE) { ntfs_error(sb, "Out of bounds check failed. Corrupt directory " "inode 0x%lx or driver bug.", dir_ni->mft_no); - err = -EIO; + goto unm_err_out; + } + /* Catch multi sector transfer fixup errors. */ + if (unlikely(!ntfs_is_indx_record(ia->magic))) { + ntfs_error(sb, "Directory index record with vcn 0x%llx is " + "corrupt. Corrupt inode 0x%lx. Run chkdsk.", + (unsigned long long)vcn, dir_ni->mft_no); goto unm_err_out; } if (sle64_to_cpu(ia->index_block_vcn) != vcn) { @@ -828,7 +832,6 @@ "bug.", (unsigned long long) sle64_to_cpu(ia->index_block_vcn), (unsigned long long)vcn, dir_ni->mft_no); - err = -EIO; goto unm_err_out; } if (le32_to_cpu(ia->index.allocated_size) + 0x18 != @@ -840,7 +843,6 @@ (unsigned long long)vcn, dir_ni->mft_no, le32_to_cpu(ia->index.allocated_size) + 0x18, dir_ni->itype.index.block_size); - err = -EIO; goto unm_err_out; } index_end = (u8*)ia + dir_ni->itype.index.block_size; @@ -850,7 +852,6 @@ "Cannot access! This is probably a bug in the " "driver.", (unsigned long long)vcn, dir_ni->mft_no); - err = -EIO; goto unm_err_out; } index_end = (u8*)&ia->index + le32_to_cpu(ia->index.index_length); @@ -858,7 +859,6 @@ ntfs_error(sb, "Size of index buffer (VCN 0x%llx) of directory " "inode 0x%lx exceeds maximum size.", (unsigned long long)vcn, dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* The first index entry. */ @@ -878,7 +878,6 @@ ntfs_error(sb, "Index entry out of bounds in " "directory inode 0x%lx.", dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* @@ -962,7 +961,6 @@ ntfs_error(sb, "Index entry with child node found in " "a leaf node in directory inode 0x%lx.", dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* Child node present, descend into it. */ @@ -982,7 +980,6 @@ } ntfs_error(sb, "Negative child node vcn in directory inode " "0x%lx.", dir_ni->mft_no); - err = -EIO; goto unm_err_out; } /* No child node, return -ENOENT. */ @@ -992,6 +989,8 @@ unlock_page(page); ntfs_unmap_page(page); err_out: + if (!err) + err = -EIO; if (ctx) ntfs_attr_put_search_ctx(ctx); if (m) @@ -999,7 +998,6 @@ return ERR_MREF(err); dir_err_out: ntfs_error(sb, "Corrupt directory. Aborting lookup."); - err = -EIO; goto err_out; } @@ -1338,6 +1336,14 @@ if (unlikely((u8*)ia < kaddr || (u8*)ia > kaddr + PAGE_CACHE_SIZE)) { ntfs_error(sb, "Out of bounds check failed. Corrupt directory " "inode 0x%lx or driver bug.", vdir->i_ino); + goto err_out; + } + /* Catch multi sector transfer fixup errors. */ + if (unlikely(!ntfs_is_indx_record(ia->magic))) { + ntfs_error(sb, "Directory index record with vcn 0x%llx is " + "corrupt. Corrupt inode 0x%lx. Run chkdsk.", + (unsigned long long)ia_pos >> + ndir->itype.index.vcn_size_bits, vdir->i_ino); goto err_out; } if (unlikely(sle64_to_cpu(ia->index_block_vcn) != (ia_pos & diff -Nru a/fs/ntfs/index.c b/fs/ntfs/index.c --- a/fs/ntfs/index.c 2004-11-10 13:45:28 +00:00 +++ b/fs/ntfs/index.c 2004-11-10 13:45:28 +00:00 @@ -263,7 +263,6 @@ ntfs_error(sb, "No index allocation attribute but index entry " "requires one. Inode 0x%lx is corrupt or " "driver bug.", idx_ni->mft_no); - err = -EIO; goto err_out; } /* Get the starting vcn of the index_block holding the child node. */ @@ -301,7 +300,13 @@ if ((u8*)ia < kaddr || (u8*)ia > kaddr + PAGE_CACHE_SIZE) { ntfs_error(sb, "Out of bounds check failed. Corrupt inode " "0x%lx or driver bug.", idx_ni->mft_no); - err = -EIO; + goto unm_err_out; + } + /* Catch multi sector transfer fixup errors. */ + if (unlikely(!ntfs_is_indx_record(ia->magic))) { + ntfs_error(sb, "Index record with vcn 0x%llx is corrupt. " + "Corrupt inode 0x%lx. Run chkdsk.", + (long long)vcn, idx_ni->mft_no); goto unm_err_out; } if (sle64_to_cpu(ia->index_block_vcn) != vcn) { @@ -311,7 +316,6 @@ (unsigned long long) sle64_to_cpu(ia->index_block_vcn), (unsigned long long)vcn, idx_ni->mft_no); - err = -EIO; goto unm_err_out; } if (le32_to_cpu(ia->index.allocated_size) + 0x18 != @@ -323,7 +327,6 @@ idx_ni->mft_no, le32_to_cpu(ia->index.allocated_size) + 0x18, idx_ni->itype.index.block_size); - err = -EIO; goto unm_err_out; } index_end = (u8*)ia + idx_ni->itype.index.block_size; @@ -333,7 +336,6 @@ "access! This is probably a bug in the " "driver.", (unsigned long long)vcn, idx_ni->mft_no); - err = -EIO; goto unm_err_out; } index_end = (u8*)&ia->index + le32_to_cpu(ia->index.index_length); @@ -341,7 +343,6 @@ ntfs_error(sb, "Size of index buffer (VCN 0x%llx) of inode " "0x%lx exceeds maximum size.", (unsigned long long)vcn, idx_ni->mft_no); - err = -EIO; goto unm_err_out; } /* The first index entry. */ @@ -359,11 +360,10 @@ (u8*)ie + le16_to_cpu(ie->length) > index_end) { ntfs_error(sb, "Index entry out of bounds in inode " "0x%lx.", idx_ni->mft_no); - err = -EIO; goto unm_err_out; } /* - * The last entry cannot contain a ket. It can however contain + * 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 (ie->flags & INDEX_ENTRY_END) @@ -377,7 +377,6 @@ le16_to_cpu(ie->length)) { ntfs_error(sb, "Index entry out of bounds in inode " "0x%lx.", idx_ni->mft_no); - err = -EIO; goto unm_err_out; } /* If the keys match perfectly, we setup @ictx and return 0. */ @@ -424,7 +423,6 @@ if ((ia->index.flags & NODE_MASK) == LEAF_NODE) { ntfs_error(sb, "Index entry with child node found in a leaf " "node in inode 0x%lx.", idx_ni->mft_no); - err = -EIO; goto unm_err_out; } /* Child node present, descend into it. */ @@ -446,11 +444,12 @@ } ntfs_error(sb, "Negative child node vcn in inode 0x%lx.", idx_ni->mft_no); - err = -EIO; unm_err_out: unlock_page(page); ntfs_unmap_page(page); err_out: + if (!err) + err = -EIO; if (actx) ntfs_attr_put_search_ctx(actx); if (m) @@ -458,6 +457,5 @@ return err; idx_err_out: ntfs_error(sb, "Corrupt index. Aborting lookup."); - err = -EIO; goto err_out; } diff -Nru a/fs/ntfs/mft.c b/fs/ntfs/mft.c --- a/fs/ntfs/mft.c 2004-11-10 13:45:28 +00:00 +++ b/fs/ntfs/mft.c 2004-11-10 13:45:28 +00:00 @@ -68,20 +68,31 @@ if (index > end_index || (mft_vi->i_size & ~PAGE_CACHE_MASK) < ofs + vol->mft_record_size) { page = ERR_PTR(-ENOENT); + ntfs_error(vol->sb, "Attemt to read mft record 0x%lx, " + "which is beyond the end of the mft. " + "This is probably a bug in the ntfs " + "driver.", ni->mft_no); goto err_out; } } /* Read, map, and pin the page. */ page = ntfs_map_page(mft_vi->i_mapping, index); if (likely(!IS_ERR(page))) { - ni->page = page; - ni->page_ofs = ofs; - return page_address(page) + ofs; + /* Catch multi sector transfer fixup errors. */ + if (likely(ntfs_is_mft_recordp((le32*)(page_address(page) + + ofs)))) { + ni->page = page; + ni->page_ofs = ofs; + return page_address(page) + ofs; + } + ntfs_error(vol->sb, "Mft record 0x%lx is corrupt. " + "Run chkdsk.", ni->mft_no); + ntfs_unmap_page(page); + page = ERR_PTR(-EIO); } err_out: ni->page = NULL; ni->page_ofs = 0; - ntfs_error(vol->sb, "Failed with error code %lu.", -PTR_ERR(page)); return (void*)page; } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:31
|
This is patch 18/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/04 1.2026.1.53) NTFS: Drop the runlist lock after the vcn has been read in fs/ntfs/lcnalloc.c::__ntfs_cluster_free(). Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:24 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:24 +00:00 @@ -70,6 +70,8 @@ error handling code path that resulted in a BUG() due to trying to unmap an extent mft record when the mapping of it had failed and it thus was not mapped. (Thanks to Ken MacFerrin for the bug report.) + - Drop the runlist lock after the vcn has been read in + fs/ntfs/lcnalloc.c::__ntfs_cluster_free(). 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/lcnalloc.c b/fs/ntfs/lcnalloc.c --- a/fs/ntfs/lcnalloc.c 2004-11-10 13:45:24 +00:00 +++ b/fs/ntfs/lcnalloc.c 2004-11-10 13:45:24 +00:00 @@ -903,8 +903,8 @@ * Attempt to map runlist, dropping runlist lock for * the duration. */ - up_read(&ni->runlist.lock); vcn = rl->vcn; + up_read(&ni->runlist.lock); err = ntfs_map_runlist(ni, vcn); if (err) { if (!is_rollback) |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:28
|
This is patch 17/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/04 1.2026.1.52) NTFS: Fix stupid bug in fs/ntfs/attrib.c::ntfs_external_attr_find() in the error handling code path that resulted in a BUG() due to trying to unmap an extent mft record when the mapping of it had failed and it thus was not mapped. (Thanks to Ken MacFerrin for the bug report.) Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:20 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:20 +00:00 @@ -66,6 +66,10 @@ buffers belonging to the ntfs record dirty. This causes the buffers to become busy and hence they are safe from removal until the page has been written out. + - Fix stupid bug in fs/ntfs/attrib.c::ntfs_external_attr_find() in the + error handling code path that resulted in a BUG() due to trying to + unmap an extent mft record when the mapping of it had failed and it + thus was not mapped. (Thanks to Ken MacFerrin for the bug report.) 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/attrib.c b/fs/ntfs/attrib.c --- a/fs/ntfs/attrib.c 2004-11-10 13:45:20 +00:00 +++ b/fs/ntfs/attrib.c 2004-11-10 13:45:20 +00:00 @@ -655,7 +655,6 @@ ctx->mrec = map_extent_mft_record(base_ni, le64_to_cpu( al_entry->mft_reference), &ni); - ctx->ntfs_ino = ni; if (IS_ERR(ctx->mrec)) { ntfs_error(vol->sb, "Failed to map " "extent mft record " @@ -667,8 +666,11 @@ err = PTR_ERR(ctx->mrec); if (err == -ENOENT) err = -EIO; + /* Cause @ctx to be sanitized below. */ + ni = NULL; break; } + ctx->ntfs_ino = ni; } ctx->attr = (ATTR_RECORD*)((u8*)ctx->mrec + le16_to_cpu(ctx->mrec->attrs_offset)); @@ -740,7 +742,8 @@ err = -EIO; } if (ni != base_ni) { - unmap_extent_mft_record(ni); + if (ni) + unmap_extent_mft_record(ni); ctx->ntfs_ino = base_ni; ctx->mrec = ctx->base_mrec; ctx->attr = ctx->base_attr; |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:25
|
This is patch 15/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/01 1.2026.53.1) NTFS: Make several functions and variables static. (Adrian Bunk) Signed-off-by: Adrian Bunk <bu...@st...> Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:12 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:12 +00:00 @@ -56,6 +56,11 @@ ntfs_prepare_write(). - Remove unused function fs/ntfs/runlist.c::ntfs_rl_merge(). (Adrian Bunk) + - Fix stupid bug in fs/ntfs/attrib.c::ntfs_attr_find() that resulted in + a NULL pointer dereference in the error code path when a corrupt + attribute was found. (Thanks to Domen Puncer for the bug report.) + - Add MODULE_VERSION() to fs/ntfs/super.c. + - Make several functions and variables static. (Adrian Bunk) 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:45:12 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:45:12 +00:00 @@ -348,10 +348,8 @@ * for it to be read in before we can do the copy. * * Return 0 on success and -errno on error. - * - * WARNING: Do not make this function static! It is used by mft.c! */ -int ntfs_readpage(struct file *file, struct page *page) +static int ntfs_readpage(struct file *file, struct page *page) { s64 attr_pos; ntfs_inode *ni, *base_ni; diff -Nru a/fs/ntfs/inode.c b/fs/ntfs/inode.c --- a/fs/ntfs/inode.c 2004-11-10 13:45:12 +00:00 +++ b/fs/ntfs/inode.c 2004-11-10 13:45:12 +00:00 @@ -352,7 +352,7 @@ return NULL; } -void ntfs_destroy_extent_inode(ntfs_inode *ni) +static void ntfs_destroy_extent_inode(ntfs_inode *ni) { ntfs_debug("Entering."); BUG_ON(ni->page); @@ -2133,7 +2133,7 @@ } } -void __ntfs_clear_inode(ntfs_inode *ni) +static void __ntfs_clear_inode(ntfs_inode *ni) { /* Free all alocated memory. */ down_write(&ni->runlist.lock); diff -Nru a/fs/ntfs/ntfs.h b/fs/ntfs/ntfs.h --- a/fs/ntfs/ntfs.h 2004-11-10 13:45:12 +00:00 +++ b/fs/ntfs/ntfs.h 2004-11-10 13:45:12 +00:00 @@ -53,7 +53,6 @@ extern kmem_cache_t *ntfs_index_ctx_cache; /* The various operations structs defined throughout the driver files. */ -extern struct super_operations ntfs_sops; extern struct address_space_operations ntfs_aops; extern struct address_space_operations ntfs_mst_aops; @@ -86,8 +85,6 @@ /* From fs/ntfs/super.c */ #define default_upcase_len 0x10000 -extern ntfschar *default_upcase; -extern unsigned long ntfs_nr_upcase_users; extern struct semaphore ntfs_lock; typedef struct { diff -Nru a/fs/ntfs/super.c b/fs/ntfs/super.c --- a/fs/ntfs/super.c 2004-11-10 13:45:12 +00:00 +++ b/fs/ntfs/super.c 2004-11-10 13:45:12 +00:00 @@ -44,6 +44,10 @@ /* Number of mounted file systems which have compression enabled. */ static unsigned long ntfs_nr_compression_users; +/* A global default upcase table and a corresponding reference count. */ +static ntfschar *default_upcase = NULL; +static unsigned long ntfs_nr_upcase_users = 0; + /* Error constants/strings used in inode.c::ntfs_show_options(). */ typedef enum { /* One of these must be present, default is ON_ERRORS_CONTINUE. */ @@ -2175,7 +2179,7 @@ /** * The complete super operations. */ -struct super_operations ntfs_sops = { +static struct super_operations ntfs_sops = { .alloc_inode = ntfs_alloc_big_inode, /* VFS: Allocate new inode. */ .destroy_inode = ntfs_destroy_big_inode, /* VFS: Deallocate inode. */ .put_inode = ntfs_put_inode, /* VFS: Called just before @@ -2592,10 +2596,6 @@ */ kmem_cache_t *ntfs_attr_ctx_cache; kmem_cache_t *ntfs_index_ctx_cache; - -/* A global default upcase table and a corresponding reference count. */ -ntfschar *default_upcase = NULL; -unsigned long ntfs_nr_upcase_users = 0; /* Driver wide semaphore. */ DECLARE_MUTEX(ntfs_lock); |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:22
|
This is patch 16/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/03 1.2026.1.50) NTFS: Modify fs/ntfs/aops.c::mark_ntfs_record_dirty() so it allocates buffers for the page if they are not present and then marks the buffers belonging to the ntfs record dirty. This causes the buffers to become busy and hence they are safe from removal until the page has been written out. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:16 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:16 +00:00 @@ -61,6 +61,11 @@ attribute was found. (Thanks to Domen Puncer for the bug report.) - Add MODULE_VERSION() to fs/ntfs/super.c. - Make several functions and variables static. (Adrian Bunk) + - Modify fs/ntfs/aops.c::mark_ntfs_record_dirty() so it allocates + buffers for the page if they are not present and then marks the + buffers belonging to the ntfs record dirty. This causes the buffers + to become busy and hence they are safe from removal until the page + has been written out. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:45:16 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:45:16 +00:00 @@ -2170,29 +2170,43 @@ * @page: page containing the ntfs record to mark dirty * @ofs: byte offset within @page at which the ntfs record begins * - * If the ntfs record is the same size as the page cache page @page, set all - * buffers in the page dirty. Otherwise, set only the buffers in which the - * ntfs record is located dirty. + * Set the buffers and the page in which the ntfs record is located dirty. * - * Also, set the page containing the ntfs record dirty, which also marks the - * vfs inode the ntfs record belongs to dirty (I_DIRTY_PAGES). + * The latter also marks the vfs inode the ntfs record belongs to dirty + * (I_DIRTY_PAGES only). + * + * If the page does not have buffers, we create them and set them uptodate. + * The page may not be locked which is why we need to handle the buffers under + * the mapping->private_lock. Once the buffers are marked dirty we no longer + * need the lock since try_to_free_buffers() does not free dirty buffers. */ void mark_ntfs_record_dirty(struct page *page, const unsigned int ofs) { - ntfs_inode *ni; - struct buffer_head *bh, *head; + struct address_space *mapping = page->mapping; + ntfs_inode *ni = NTFS_I(mapping->host); + struct buffer_head *bh, *head, *buffers_to_free = NULL; unsigned int end, bh_size, bh_ofs; - BUG_ON(!page); - BUG_ON(!page_has_buffers(page)); - ni = NTFS_I(page->mapping->host); - BUG_ON(!ni); - if (ni->itype.index.block_size == PAGE_CACHE_SIZE) { - __set_page_dirty_buffers(page); - return; - } + BUG_ON(!PageUptodate(page)); end = ofs + ni->itype.index.block_size; - bh_size = ni->vol->sb->s_blocksize; - spin_lock(&page->mapping->private_lock); + bh_size = 1 << VFS_I(ni)->i_blkbits; + spin_lock(&mapping->private_lock); + if (unlikely(!page_has_buffers(page))) { + spin_unlock(&mapping->private_lock); + bh = head = alloc_page_buffers(page, bh_size, 1); + spin_lock(&mapping->private_lock); + if (likely(!page_has_buffers(page))) { + struct buffer_head *tail; + + do { + set_buffer_uptodate(bh); + tail = bh; + bh = bh->b_this_page; + } while (bh); + tail->b_this_page = head; + attach_page_buffers(page, head); + } else + buffers_to_free = bh; + } bh = head = page_buffers(page); do { bh_ofs = bh_offset(bh); @@ -2202,8 +2216,15 @@ break; set_buffer_dirty(bh); } while ((bh = bh->b_this_page) != head); - spin_unlock(&page->mapping->private_lock); + spin_unlock(&mapping->private_lock); __set_page_dirty_nobuffers(page); + if (unlikely(buffers_to_free)) { + do { + bh = buffers_to_free->b_this_page; + free_buffer_head(buffers_to_free); + buffers_to_free = bh; + } while (buffers_to_free); + } } #endif /* NTFS_RW */ |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:13
|
This is patch 13/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/11/01 1.2026.48.1) NTFS: Fix stupid bug in fs/ntfs/attrib.c::ntfs_attr_find() that resulted in a NULL pointer dereference in the error code path when a corrupt attribute was found. Thanks to Domen Puncer for the bug report. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/attrib.c b/fs/ntfs/attrib.c --- a/fs/ntfs/attrib.c 2004-11-10 13:45:05 +00:00 +++ b/fs/ntfs/attrib.c 2004-11-10 13:45:05 +00:00 @@ -250,19 +250,10 @@ const u8 *val, const u32 val_len, ntfs_attr_search_ctx *ctx) { ATTR_RECORD *a; - ntfs_volume *vol; - ntfschar *upcase; - u32 upcase_len; + ntfs_volume *vol = ctx->ntfs_ino->vol; + ntfschar *upcase = vol->upcase; + u32 upcase_len = vol->upcase_len; - if (ic == IGNORE_CASE) { - vol = ctx->ntfs_ino->vol; - upcase = vol->upcase; - upcase_len = vol->upcase_len; - } else { - vol = NULL; - upcase = NULL; - upcase_len = 0; - } /* * Iterate over attributes in mft record starting at @ctx->attr, or the * attribute following that, if @ctx->is_first is TRUE. @@ -354,7 +345,7 @@ return -ENOENT; } } - ntfs_error(NULL, "Inode is corrupt. Run chkdsk."); + ntfs_error(vol->sb, "Inode is corrupt. Run chkdsk."); NVolSetErrors(vol); return -EIO; } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:07
|
This is patch 12/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/29 1.2026.37.1) NTFS: Remove unused function fs/ntfs/runlist.c::ntfs_rl_merge(). (Adrian Bunk) Signed-off-by: Adrian Bunk <bu...@st...> Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:45:01 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:45:01 +00:00 @@ -12,6 +12,10 @@ OTOH, perhaps i_sem, which is held accross generic_file_write is sufficient for synchronisation here. We then just need to make sure ntfs_readpage/writepage/truncate interoperate properly with us. + UPDATE: The above is all ok as it is due to i_sem held. The only + thing that needs to be checked is ntfs_writepage() which does not + hold i_sem. It cannot change i_size but it needs to cope with a + concurrent i_size change. - Implement mft.c::sync_mft_mirror_umount(). We currently will just leave the volume dirty on umount if the final iput(vol->mft_ino) causes a write of any mirrored mft records due to the mft mirror @@ -42,7 +46,7 @@ mount time as this cannot work with the current implementation. - Check for location of attribute name and improve error handling in general in fs/ntfs/inode.c::ntfs_read_locked_inode() and friends. - - In fs/ntfs/aops.c::ntfs_writepage(), if t he page is fully outside + - In fs/ntfs/aops.c::ntfs_writepage(), if the page is fully outside i_size, i.e. race with truncate, invalidate the buffers on the page so that they become freeable and hence the page does not leak. - Implement extension of resident files in the regular file write code @@ -50,6 +54,8 @@ this only works until the data attribute becomes too big for the mft record after which we abort the write returning -EOPNOTSUPP from ntfs_prepare_write(). + - Remove unused function fs/ntfs/runlist.c::ntfs_rl_merge(). (Adrian + Bunk) 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/runlist.c b/fs/ntfs/runlist.c --- a/fs/ntfs/runlist.c 2004-11-10 13:45:01 +00:00 +++ b/fs/ntfs/runlist.c 2004-11-10 13:45:01 +00:00 @@ -140,30 +140,6 @@ } /** - * ntfs_rl_merge - test if two runlists can be joined together and merge them - * @dst: original, destination runlist - * @src: new runlist to merge with @dst - * - * Test if two runlists can be joined together. For this, their VCNs and LCNs - * must be adjacent. If they can be merged, perform the merge, writing into - * the destination runlist @dst. - * - * It is up to the caller to serialize access to the runlists @dst and @src. - * - * Return: TRUE Success, the runlists have been merged. - * FALSE Failure, the runlists cannot be merged and have not been - * modified. - */ -static inline BOOL ntfs_rl_merge(runlist_element *dst, runlist_element *src) -{ - BOOL merge = ntfs_are_rl_mergeable(dst, src); - - if (merge) - __ntfs_rl_merge(dst, src); - return merge; -} - -/** * ntfs_rl_append - append a runlist after a given element * @dst: original runlist to be worked on * @dsize: number of elements in @dst (including end marker) |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:05
|
This is patch 9/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/27 1.2026.1.32) NTFS: Check for location of attribute name and improve error handling in general in fs/ntfs/inode.c::ntfs_read_locked_inode() and friends. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:48 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:48 +00:00 @@ -40,6 +40,8 @@ analagous to the way it is done in __set_page_dirty_buffers(). - Ensure the mft record size does not exceed the PAGE_CACHE_SIZE at mount time as this cannot work with the current implementation. + - Check for location of attribute name and improve error handling in + general in fs/ntfs/inode.c::ntfs_read_locked_inode() and friends. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/inode.c b/fs/ntfs/inode.c --- a/fs/ntfs/inode.c 2004-11-10 13:44:48 +00:00 +++ b/fs/ntfs/inode.c 2004-11-10 13:44:48 +00:00 @@ -564,13 +564,11 @@ } if (!(m->flags & MFT_RECORD_IN_USE)) { - ntfs_error(vi->i_sb, "Inode is not in use! You should " - "run chkdsk."); + ntfs_error(vi->i_sb, "Inode is not in use!"); goto unm_err_out; } if (m->base_mft_record) { - ntfs_error(vi->i_sb, "Inode is an extent inode! You should " - "run chkdsk."); + ntfs_error(vi->i_sb, "Inode is an extent inode!"); goto unm_err_out; } @@ -667,7 +665,7 @@ if (err) { if (unlikely(err != -ENOENT)) { ntfs_error(vi->i_sb, "Failed to lookup attribute list " - "attribute. You should run chkdsk."); + "attribute."); goto unm_err_out; } } else /* if (!err) */ { @@ -679,9 +677,7 @@ ctx->attr->flags & ATTR_COMPRESSION_MASK || ctx->attr->flags & ATTR_IS_SPARSE) { ntfs_error(vi->i_sb, "Attribute list attribute is " - "compressed/encrypted/sparse. Not " - "allowed. Corrupt inode. You should " - "run chkdsk."); + "compressed/encrypted/sparse."); goto unm_err_out; } /* Now allocate memory for the attribute list. */ @@ -697,9 +693,7 @@ NInoSetAttrListNonResident(ni); if (ctx->attr->data.non_resident.lowest_vcn) { ntfs_error(vi->i_sb, "Attribute list has non " - "zero lowest_vcn. Inode is " - "corrupt. You should run " - "chkdsk."); + "zero lowest_vcn."); goto unm_err_out; } /* @@ -712,10 +706,7 @@ err = PTR_ERR(ni->attr_list_rl.rl); ni->attr_list_rl.rl = NULL; ntfs_error(vi->i_sb, "Mapping pairs " - "decompression failed with " - "error code %i. Corrupt " - "attribute list in inode.", - -err); + "decompression failed."); goto unm_err_out; } /* Now load the attribute list. */ @@ -770,9 +761,18 @@ goto unm_err_out; } /* Set up the state. */ - if (ctx->attr->non_resident) { - ntfs_error(vi->i_sb, "$INDEX_ROOT attribute is " - "not resident. Not allowed."); + if (unlikely(ctx->attr->non_resident)) { + ntfs_error(vol->sb, "$INDEX_ROOT attribute is not " + "resident."); + goto unm_err_out; + } + /* Ensure the attribute name is placed before the value. */ + if (unlikely(ctx->attr->name_length && + (le16_to_cpu(ctx->attr->name_offset) >= + le16_to_cpu(ctx->attr->data.resident. + value_offset)))) { + ntfs_error(vol->sb, "$INDEX_ROOT attribute name is " + "placed after the attribute value."); goto unm_err_out; } /* @@ -786,8 +786,7 @@ if (ctx->attr->flags & ATTR_IS_ENCRYPTED) { if (ctx->attr->flags & ATTR_COMPRESSION_MASK) { ntfs_error(vi->i_sb, "Found encrypted and " - "compressed attribute. Not " - "allowed."); + "compressed attribute."); goto unm_err_out; } NInoSetEncrypted(ni); @@ -811,12 +810,12 @@ } if (ir->type != AT_FILE_NAME) { ntfs_error(vi->i_sb, "Indexed attribute is not " - "$FILE_NAME. Not allowed."); + "$FILE_NAME."); goto unm_err_out; } if (ir->collation_rule != COLLATION_FILE_NAME) { ntfs_error(vi->i_sb, "Index collation rule is not " - "COLLATION_FILE_NAME. Not allowed."); + "COLLATION_FILE_NAME."); goto unm_err_out; } ni->itype.index.collation_rule = ir->collation_rule; @@ -831,7 +830,7 @@ if (ni->itype.index.block_size > PAGE_CACHE_SIZE) { ntfs_error(vi->i_sb, "Index block size (%u) > " "PAGE_CACHE_SIZE (%ld) is not " - "supported. Sorry.", + "supported. Sorry.", ni->itype.index.block_size, PAGE_CACHE_SIZE); err = -EOPNOTSUPP; @@ -840,7 +839,7 @@ if (ni->itype.index.block_size < NTFS_BLOCK_SIZE) { ntfs_error(vi->i_sb, "Index block size (%u) < " "NTFS_BLOCK_SIZE (%i) is not " - "supported. Sorry.", + "supported. Sorry.", ni->itype.index.block_size, NTFS_BLOCK_SIZE); err = -EOPNOTSUPP; @@ -883,8 +882,7 @@ if (err == -ENOENT) ntfs_error(vi->i_sb, "$INDEX_ALLOCATION " "attribute is not present but " - "$INDEX_ROOT indicated it " - "is."); + "$INDEX_ROOT indicated it is."); else ntfs_error(vi->i_sb, "Failed to lookup " "$INDEX_ALLOCATION " @@ -896,6 +894,19 @@ "is resident."); goto unm_err_out; } + /* + * Ensure the attribute name is placed before the mapping pairs + * array. + */ + if (unlikely(ctx->attr->name_length && + (le16_to_cpu(ctx->attr->name_offset) >= + le16_to_cpu(ctx->attr->data.non_resident. + mapping_pairs_offset)))) { + ntfs_error(vol->sb, "$INDEX_ALLOCATION attribute name " + "is placed after the mapping pairs " + "array."); + goto unm_err_out; + } if (ctx->attr->flags & ATTR_IS_ENCRYPTED) { ntfs_error(vi->i_sb, "$INDEX_ALLOCATION attribute " "is encrypted."); @@ -914,8 +925,7 @@ if (ctx->attr->data.non_resident.lowest_vcn) { ntfs_error(vi->i_sb, "First extent of " "$INDEX_ALLOCATION attribute has non " - "zero lowest_vcn. Inode is corrupt. " - "You should run chkdsk."); + "zero lowest_vcn."); goto unm_err_out; } vi->i_size = sle64_to_cpu( @@ -997,8 +1007,7 @@ goto no_data_attr_special_case; // FIXME: File is corrupt! Hot-fix with empty data // attribute if recovery option is set. - ntfs_error(vi->i_sb, "$DATA attribute is " - "missing."); + ntfs_error(vi->i_sb, "$DATA attribute is missing."); goto unm_err_out; } /* Setup the state. */ @@ -1028,10 +1037,8 @@ compression_unit != 4) { ntfs_error(vi->i_sb, "Found " "nonstandard compression unit " - "(%u instead of 4). Cannot " - "handle this. This might " - "indicate corruption so you " - "should run chkdsk.", + "(%u instead of 4). Cannot " + "handle this.", ctx->attr->data.non_resident. compression_unit); err = -EOPNOTSUPP; @@ -1057,8 +1064,7 @@ if (ctx->attr->data.non_resident.lowest_vcn) { ntfs_error(vi->i_sb, "First extent of $DATA " "attribute has non zero " - "lowest_vcn. Inode is corrupt. " - "You should run chkdsk."); + "lowest_vcn."); goto unm_err_out; } /* Setup all the sizes. */ @@ -1127,9 +1133,11 @@ if (m) unmap_mft_record(ni); err_out: - ntfs_error(vi->i_sb, "Failed with error code %i. Marking inode 0x%lx " - "as bad.", -err, vi->i_ino); + ntfs_error(vol->sb, "Failed with error code %i. Marking corrupt " + "inode 0x%lx as bad. Run chkdsk.", err, vi->i_ino); make_bad_inode(vi); + if (err != -EOPNOTSUPP && err != -ENOMEM) + NVolSetErrors(vol); return err; } @@ -1200,15 +1208,21 @@ goto unm_err_out; if (!ctx->attr->non_resident) { + /* Ensure the attribute name is placed before the value. */ + if (unlikely(ctx->attr->name_length && + (le16_to_cpu(ctx->attr->name_offset) >= + le16_to_cpu(ctx->attr->data.resident. + value_offset)))) { + ntfs_error(vol->sb, "Attribute name is placed after " + "the attribute value."); + goto unm_err_out; + } if (NInoMstProtected(ni) || ctx->attr->flags) { ntfs_error(vi->i_sb, "Found mst protected attribute " "or attribute with non-zero flags but " - "the attribute is resident (mft_no " - "0x%lx, type 0x%x, name_len %i). " - "Please report you saw this message " - "to linux-ntfs-dev@lists." - "sourceforge.net", - vi->i_ino, ni->type, ni->name_len); + "the attribute is resident. Please " + "report you saw this message to " + "lin...@li..."); goto unm_err_out; } /* @@ -1219,60 +1233,63 @@ le32_to_cpu(ctx->attr->data.resident.value_length); } else { NInoSetNonResident(ni); + /* + * Ensure the attribute name is placed before the mapping pairs + * array. + */ + if (unlikely(ctx->attr->name_length && + (le16_to_cpu(ctx->attr->name_offset) >= + le16_to_cpu(ctx->attr->data.non_resident. + mapping_pairs_offset)))) { + ntfs_error(vol->sb, "Attribute name is placed after " + "the mapping pairs array."); + goto unm_err_out; + } if (ctx->attr->flags & ATTR_COMPRESSION_MASK) { if (NInoMstProtected(ni)) { ntfs_error(vi->i_sb, "Found mst protected " "attribute but the attribute " - "is compressed (mft_no 0x%lx, " - "type 0x%x, name_len %i). " - "Please report you saw this " - "message to linux-ntfs-dev@" - "lists.sourceforge.net", - vi->i_ino, ni->type, - ni->name_len); + "is compressed. Please report " + "you saw this message to " + "linux-ntfs-dev@lists." + "sourceforge.net"); goto unm_err_out; } NInoSetCompressed(ni); if ((ni->type != AT_DATA) || (ni->type == AT_DATA && ni->name_len)) { - ntfs_error(vi->i_sb, "Found compressed non-" - "data or named data attribute " - "(mft_no 0x%lx, type 0x%x, " - "name_len %i). Please report " + ntfs_error(vi->i_sb, "Found compressed " + "non-data or named data " + "attribute. Please report " "you saw this message to " "linux-ntfs-dev@lists." - "sourceforge.net", - vi->i_ino, ni->type, - ni->name_len); + "sourceforge.net"); goto unm_err_out; } if (vol->cluster_size > 4096) { - ntfs_error(vi->i_sb, "Found " - "compressed attribute but " - "compression is disabled due " - "to cluster size (%i) > 4kiB.", - vol->cluster_size); + ntfs_error(vi->i_sb, "Found compressed " + "attribute but compression is " + "disabled due to cluster size " + "(%i) > 4kiB.", + vol->cluster_size); goto unm_err_out; } if ((ctx->attr->flags & ATTR_COMPRESSION_MASK) != ATTR_IS_COMPRESSED) { ntfs_error(vi->i_sb, "Found unknown " - "compression method or " - "corrupt file."); + "compression method."); goto unm_err_out; } ni->itype.compressed.block_clusters = 1U << ctx->attr->data.non_resident. compression_unit; - if (ctx->attr->data.non_resident.compression_unit != 4) { - ntfs_error(vi->i_sb, "Found " - "nonstandard compression unit " - "(%u instead of 4). Cannot " - "handle this. This might " - "indicate corruption so you " - "should run chkdsk.", - ctx->attr->data.non_resident. - compression_unit); + if (ctx->attr->data.non_resident.compression_unit != + 4) { + ntfs_error(vi->i_sb, "Found nonstandard " + "compression unit (%u instead " + "of 4). Cannot handle this.", + ctx->attr->data.non_resident. + compression_unit); err = -EOPNOTSUPP; goto unm_err_out; } @@ -1292,13 +1309,10 @@ if (NInoMstProtected(ni)) { ntfs_error(vi->i_sb, "Found mst protected " "attribute but the attribute " - "is encrypted (mft_no 0x%lx, " - "type 0x%x, name_len %i). " - "Please report you saw this " - "message to linux-ntfs-dev@" - "lists.sourceforge.net", - vi->i_ino, ni->type, - ni->name_len); + "is encrypted. Please report " + "you saw this message to " + "linux-ntfs-dev@lists." + "sourceforge.net"); goto unm_err_out; } NInoSetEncrypted(ni); @@ -1307,21 +1321,17 @@ if (NInoMstProtected(ni)) { ntfs_error(vi->i_sb, "Found mst protected " "attribute but the attribute " - "is sparse (mft_no 0x%lx, " - "type 0x%x, name_len %i). " - "Please report you saw this " - "message to linux-ntfs-dev@" - "lists.sourceforge.net", - vi->i_ino, ni->type, - ni->name_len); + "is sparse. Please report " + "you saw this message to " + "linux-ntfs-dev@lists." + "sourceforge.net"); goto unm_err_out; } NInoSetSparse(ni); } if (ctx->attr->data.non_resident.lowest_vcn) { ntfs_error(vi->i_sb, "First extent of attribute has " - "non-zero lowest_vcn. Inode is " - "corrupt. You should run chkdsk."); + "non-zero lowest_vcn."); goto unm_err_out; } /* Setup all the sizes. */ @@ -1372,10 +1382,15 @@ ntfs_attr_put_search_ctx(ctx); unmap_mft_record(base_ni); err_out: - ntfs_error(vi->i_sb, "Failed with error code %i while reading " - "attribute inode (mft_no 0x%lx, type 0x%x, name_len " - "%i.", -err, vi->i_ino, ni->type, ni->name_len); + ntfs_error(vol->sb, "Failed with error code %i while reading attribute " + "inode (mft_no 0x%lx, type 0x%x, name_len %i). " + "Marking corrupt inode and base inode 0x%lx as bad. " + "Run chkdsk.", err, vi->i_ino, ni->type, ni->name_len, + base_vi->i_ino); make_bad_inode(vi); + make_bad_inode(base_vi); + if (err != -ENOMEM) + NVolSetErrors(vol); return err; } @@ -1460,16 +1475,24 @@ goto unm_err_out; } /* Set up the state. */ - if (ctx->attr->non_resident) { - ntfs_error(vi->i_sb, "$INDEX_ROOT attribute is not resident. " - "Not allowed."); + if (unlikely(ctx->attr->non_resident)) { + ntfs_error(vol->sb, "$INDEX_ROOT attribute is not resident."); + goto unm_err_out; + } + /* Ensure the attribute name is placed before the value. */ + if (unlikely(ctx->attr->name_length && + (le16_to_cpu(ctx->attr->name_offset) >= + le16_to_cpu(ctx->attr->data.resident. + value_offset)))) { + ntfs_error(vol->sb, "$INDEX_ROOT attribute name is placed " + "after the attribute value."); goto unm_err_out; } /* Compressed/encrypted/sparse index root is not allowed. */ if (ctx->attr->flags & (ATTR_COMPRESSION_MASK | ATTR_IS_ENCRYPTED | ATTR_IS_SPARSE)) { ntfs_error(vi->i_sb, "Found compressed/encrypted/sparse index " - "root attribute. Not allowed."); + "root attribute."); goto unm_err_out; } ir = (INDEX_ROOT*)((u8*)ctx->attr + @@ -1485,8 +1508,8 @@ goto unm_err_out; } if (ir->type) { - ntfs_error(vi->i_sb, "Index type is not 0 (type is 0x%x). " - "Not allowed.", le32_to_cpu(ir->type)); + ntfs_error(vi->i_sb, "Index type is not 0 (type is 0x%x).", + le32_to_cpu(ir->type)); goto unm_err_out; } ni->itype.index.collation_rule = ir->collation_rule; @@ -1552,6 +1575,16 @@ "resident."); goto unm_err_out; } + /* + * Ensure the attribute name is placed before the mapping pairs array. + */ + if (unlikely(ctx->attr->name_length && (le16_to_cpu( + ctx->attr->name_offset) >= le16_to_cpu( + ctx->attr->data.non_resident.mapping_pairs_offset)))) { + ntfs_error(vol->sb, "$INDEX_ALLOCATION attribute name is " + "placed after the mapping pairs array."); + goto unm_err_out; + } if (ctx->attr->flags & ATTR_IS_ENCRYPTED) { ntfs_error(vi->i_sb, "$INDEX_ALLOCATION attribute is " "encrypted."); @@ -1568,8 +1601,7 @@ } if (ctx->attr->data.non_resident.lowest_vcn) { ntfs_error(vi->i_sb, "First extent of $INDEX_ALLOCATION " - "attribute has non zero lowest_vcn. Inode is " - "corrupt. You should run chkdsk."); + "attribute has non zero lowest_vcn."); goto unm_err_out; } vi->i_size = sle64_to_cpu(ctx->attr->data.non_resident.data_size); @@ -1595,16 +1627,16 @@ bni = NTFS_I(bvi); if (NInoCompressed(bni) || NInoEncrypted(bni) || NInoSparse(bni)) { - ntfs_error(vi->i_sb, "$BITMAP attribute is compressed " - "and/or encrypted and/or sparse."); + ntfs_error(vi->i_sb, "$BITMAP attribute is compressed and/or " + "encrypted and/or sparse."); goto iput_unm_err_out; } /* Consistency check bitmap size vs. index allocation size. */ if ((bvi->i_size << 3) < (vi->i_size >> ni->itype.index.block_size_bits)) { - ntfs_error(vi->i_sb, "Index bitmap too small (0x%llx) " - "for index allocation (0x%llx).", - bvi->i_size << 3, vi->i_size); + ntfs_error(vi->i_sb, "Index bitmap too small (0x%llx) for " + "index allocation (0x%llx).", bvi->i_size << 3, + vi->i_size); goto iput_unm_err_out; } ni->itype.index.bmp_ino = bvi; @@ -1637,9 +1669,11 @@ unmap_mft_record(base_ni); err_out: ntfs_error(vi->i_sb, "Failed with error code %i while reading index " - "inode (mft_no 0x%lx, name_len %i.", -err, vi->i_ino, + "inode (mft_no 0x%lx, name_len %i.", err, vi->i_ino, ni->name_len); make_bad_inode(vi); + if (err != -EOPNOTSUPP && err != -ENOMEM) + NVolSetErrors(vol); return err; } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:45:01
|
This is patch 11/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/28 1.2026.1.36) NTFS: Implement extension of resident files in the regular file write code paths (fs/ntfs/aops.c::ntfs_{prepare,commit}_write()). At present this only works until the data attribute becomes too big for the mft record after which we abort the write returning -EOPNOTSUPP from ntfs_prepare_write(). Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:56 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:56 +00:00 @@ -45,6 +45,11 @@ - In fs/ntfs/aops.c::ntfs_writepage(), if t he page is fully outside i_size, i.e. race with truncate, invalidate the buffers on the page so that they become freeable and hence the page does not leak. + - Implement extension of resident files in the regular file write code + paths (fs/ntfs/aops.c::ntfs_{prepare,commit}_write()). At present + this only works until the data attribute becomes too big for the mft + record after which we abort the write returning -EOPNOTSUPP from + ntfs_prepare_write(). 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:44:56 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:44:56 +00:00 @@ -1201,10 +1201,9 @@ /* Normal data stream. */ return ntfs_write_block(wbc, page); } - /* - * Attribute is resident, implying it is not compressed, encrypted, or - * mst protected. + * Attribute is resident, implying it is not compressed, encrypted, + * sparse, or mst protected. */ BUG_ON(page_has_buffers(page)); BUG_ON(!PageUptodate(page)); @@ -1283,6 +1282,9 @@ * zeroing below is enabled, we MUST move the unlock_page() from above * to after the kunmap_atomic(), i.e. just before the * end_page_writeback(). + * UPDATE: ntfs_prepare/commit_write() do the zeroing on i_size + * increases for resident attributes so those are ok. + * TODO: ntfs_truncate(), others? */ kaddr = kmap_atomic(page, KM_USER0); @@ -1360,7 +1362,6 @@ page->index, from, to); BUG_ON(!NInoNonResident(ni)); - BUG_ON(NInoMstProtected(ni)); blocksize_bits = vi->i_blkbits; blocksize = 1 << blocksize_bits; @@ -1688,8 +1689,8 @@ * ntfs_prepare_write - prepare a page for receiving data * * This is called from generic_file_write() with i_sem held on the inode - * (@page->mapping->host). The @page is locked and kmap()ped so page_address() - * can simply be used. The source data has not yet been copied into the @page. + * (@page->mapping->host). The @page is locked but not kmap()ped. The source + * data has not yet been copied into the @page. * * Need to extend the attribute/fill in holes if necessary, create blocks and * make partially overwritten blocks uptodate, @@ -1699,8 +1700,8 @@ * Return 0 on success or -errno on error. * * Should be using block_prepare_write() [support for sparse files] or - * cont_prepare_write() [no support for sparse files]. Can't do that due to - * ntfs specifics but can look at them for implementation guidancea. + * cont_prepare_write() [no support for sparse files]. Cannot do that due to + * ntfs specifics but can look at them for implementation guidance. * * Note: In the range, @from is inclusive and @to is exclusive, i.e. @from is * the first byte in the page that will be written to and @to is the first byte @@ -1709,18 +1710,40 @@ static int ntfs_prepare_write(struct file *file, struct page *page, unsigned from, unsigned to) { + s64 new_size; struct inode *vi = page->mapping->host; - ntfs_inode *ni = NTFS_I(vi); + ntfs_inode *base_ni = NULL, *ni = NTFS_I(vi); + ntfs_volume *vol = ni->vol; + ntfs_attr_search_ctx *ctx = NULL; + MFT_RECORD *m = NULL; + ATTR_RECORD *a; + u8 *kaddr; + u32 attr_len; + int err; ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index " "0x%lx, from = %u, to = %u.", vi->i_ino, ni->type, page->index, from, to); - BUG_ON(!PageLocked(page)); BUG_ON(from > PAGE_CACHE_SIZE); BUG_ON(to > PAGE_CACHE_SIZE); BUG_ON(from > to); - + BUG_ON(NInoMstProtected(ni)); + /* + * If a previous ntfs_truncate() failed, repeat it and abort if it + * fails again. + */ + if (unlikely(NInoTruncateFailed(ni))) { + down_write(&vi->i_alloc_sem); + err = ntfs_truncate(vi); + up_write(&vi->i_alloc_sem); + if (err || NInoTruncateFailed(ni)) { + if (!err) + err = -EIO; + goto err_out; + } + } + /* If the attribute is not resident, deal with it elsewhere. */ if (NInoNonResident(ni)) { /* * Only unnamed $DATA attributes can be compressed, encrypted, @@ -1749,33 +1772,106 @@ return -EOPNOTSUPP; } } - - // TODO: Implement and remove this check. - if (NInoMstProtected(ni)) { - ntfs_error(vi->i_sb, "Writing to MST protected " - "attributes is not supported yet. " - "Sorry."); - return -EOPNOTSUPP; - } - /* Normal data stream. */ return ntfs_prepare_nonresident_write(page, from, to); } - /* * Attribute is resident, implying it is not compressed, encrypted, or - * mst protected. + * sparse. */ BUG_ON(page_has_buffers(page)); - - /* Do we need to resize the attribute? */ - if (((s64)page->index << PAGE_CACHE_SHIFT) + to > vi->i_size) { - // TODO: Implement resize... - ntfs_error(vi->i_sb, "Writing beyond the existing file size is " - "not supported yet. Sorry."); - return -EOPNOTSUPP; + new_size = ((s64)page->index << PAGE_CACHE_SHIFT) + to; + /* If we do not need to resize the attribute allocation we are done. */ + if (new_size <= vi->i_size) + goto done; + /* Map, pin, and lock the (base) mft record. */ + if (!NInoAttr(ni)) + base_ni = ni; + else + base_ni = ni->ext.base_ntfs_ino; + m = map_mft_record(base_ni); + if (IS_ERR(m)) { + err = PTR_ERR(m); + m = NULL; + ctx = NULL; + goto err_out; } - + ctx = ntfs_attr_get_search_ctx(base_ni, m); + if (unlikely(!ctx)) { + err = -ENOMEM; + goto err_out; + } + err = ntfs_attr_lookup(ni->type, ni->name, ni->name_len, + CASE_SENSITIVE, 0, NULL, 0, ctx); + if (unlikely(err)) { + if (err == -ENOENT) + err = -EIO; + goto err_out; + } + m = ctx->mrec; + a = ctx->attr; + /* The total length of the attribute value. */ + attr_len = le32_to_cpu(a->data.resident.value_length); + BUG_ON(vi->i_size != attr_len); + /* Check if new size is allowed in $AttrDef. */ + err = ntfs_attr_size_bounds_check(vol, ni->type, new_size); + if (unlikely(err)) { + if (err == -ERANGE) { + ntfs_error(vol->sb, "Write would cause the inode " + "0x%lx to exceed the maximum size for " + "its attribute type (0x%x). Aborting " + "write.", vi->i_ino, + le32_to_cpu(ni->type)); + } else { + ntfs_error(vol->sb, "Inode 0x%lx has unknown " + "attribute type 0x%x. Aborting " + "write.", vi->i_ino, + le32_to_cpu(ni->type)); + err = -EIO; + } + goto err_out2; + } + /* + * Extend the attribute record to be able to store the new attribute + * size. + */ + if (new_size >= vol->mft_record_size || ntfs_attr_record_resize(m, a, + le16_to_cpu(a->data.resident.value_offset) + + new_size)) { + /* Not enough space in the mft record. */ + ntfs_error(vol->sb, "Not enough space in the mft record for " + "the resized attribute value. This is not " + "supported yet. Aborting write."); + err = -EOPNOTSUPP; + goto err_out2; + } + /* + * We have enough space in the mft record to fit the write. This + * implies the attribute is smaller than the mft record and hence the + * attribute must be in a single page and hence page->index must be 0. + */ + BUG_ON(page->index); + /* + * If the beginning of the write is past the old size, enlarge the + * attribute value up to the beginning of the write and fill it with + * zeroes. + */ + if (from > attr_len) { + memset((u8*)a + le16_to_cpu(a->data.resident.value_offset) + + attr_len, 0, from - attr_len); + a->data.resident.value_length = cpu_to_le32(from); + /* Zero the corresponding area in the page as well. */ + if (PageUptodate(page)) { + kaddr = kmap_atomic(page, KM_USER0); + memset(kaddr + attr_len, 0, from - attr_len); + kunmap_atomic(kaddr, KM_USER0); + flush_dcache_page(page); + } + } + flush_dcache_mft_record_page(ctx->ntfs_ino); + mark_mft_record_dirty(ctx->ntfs_ino); + ntfs_attr_put_search_ctx(ctx); + unmap_mft_record(base_ni); /* * Because resident attributes are handled by memcpy() to/from the * corresponding MFT record, and because this form of i/o is byte @@ -1785,26 +1881,30 @@ * generic_file_write() does the copying from userspace. * * We thus defer the uptodate bringing of the page region outside the - * region written to to ntfs_commit_write(). The reason for doing this - * is that we save one round of: - * map_mft_record(), ntfs_attr_get_search_ctx(), - * ntfs_attr_lookup(), kmap_atomic(), kunmap_atomic(), - * ntfs_attr_put_search_ctx(), unmap_mft_record(). - * Which is obviously a very worthwhile save. - * - * Thus we just return success now... + * region written to to ntfs_commit_write(), which makes the code + * simpler and saves one atomic kmap which is good. */ +done: ntfs_debug("Done."); return 0; +err_out: + if (err == -ENOMEM) + ntfs_warning(vi->i_sb, "Error allocating memory required to " + "prepare the write."); + else { + ntfs_error(vi->i_sb, "Resident attribute prepare write failed " + "with error %i.", err); + NVolSetErrors(vol); + make_bad_inode(vi); + } +err_out2: + if (ctx) + ntfs_attr_put_search_ctx(ctx); + if (m) + unmap_mft_record(base_ni); + return err; } -/* - * NOTES: There is a disparity between the apparent need to extend the - * attribute in prepare write but to update i_size only in commit write. - * Need to make sure i_sem protection is sufficient. And if not will need to - * handle this in some way or another. - */ - /** * ntfs_commit_nonresident_write - * @@ -1813,24 +1913,21 @@ unsigned from, unsigned to) { s64 pos = ((s64)page->index << PAGE_CACHE_SHIFT) + to; - struct inode *vi; + struct inode *vi = page->mapping->host; struct buffer_head *bh, *head; unsigned int block_start, block_end, blocksize; BOOL partial; - vi = page->mapping->host; - ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index " "0x%lx, from = %u, to = %u.", vi->i_ino, NTFS_I(vi)->type, page->index, from, to); - blocksize = 1 << vi->i_blkbits; - // FIXME: We need a whole slew of special cases in here for MST - // protected attributes for example. For compressed files, too... + // FIXME: We need a whole slew of special cases in here for compressed + // files for example... // For now, we know ntfs_prepare_write() would have failed so we can't // get here in any of the cases which we have to special case, so we - // are just a ripped off unrolled generic_commit_write() at present. + // are just a ripped off, unrolled generic_commit_write(). bh = head = page_buffers(page); block_start = 0; @@ -1845,24 +1942,22 @@ mark_buffer_dirty(bh); } } while (block_start = block_end, (bh = bh->b_this_page) != head); - /* * If this is a partial write which happened to make all buffers * uptodate then we can optimize away a bogus ->readpage() for the next - * read(). Here we 'discover' whether the page went uptodate as a + * read(). Here we 'discover' whether the page went uptodate as a * result of this (potentially partial) write. */ if (!partial) SetPageUptodate(page); - /* - * Not convinced about this at all. See disparity comment above. For + * Not convinced about this at all. See disparity comment above. For * now we know ntfs_prepare_write() would have failed in the write * exceeds i_size case, so this will never trigger which is fine. */ if (pos > vi->i_size) { ntfs_error(vi->i_sb, "Writing beyond the existing file size is " - "not supported yet. Sorry."); + "not supported yet. Sorry."); return -EOPNOTSUPP; // vi->i_size = pos; // mark_inode_dirty(vi); @@ -1875,118 +1970,73 @@ * ntfs_commit_write - commit the received data * * This is called from generic_file_write() with i_sem held on the inode - * (@page->mapping->host). The @page is locked and kmap()ped so page_address() - * can simply be used. The source data has already been copied into the @page. + * (@page->mapping->host). The @page is locked but not kmap()ped. The source + * data has already been copied into the @page. ntfs_prepare_write() has been + * called before the data copied and it returned success so we can take the + * results of various BUG checks and some error handling for granted. * * Need to mark modified blocks dirty so they get written out later when * ntfs_writepage() is invoked by the VM. * * Return 0 on success or -errno on error. * - * Should be using generic_commit_write(). This marks buffers uptodate and + * Should be using generic_commit_write(). This marks buffers uptodate and * dirty, sets the page uptodate if all buffers in the page are uptodate, and - * updates i_size if the end of io is beyond i_size. In that case, it also - * marks the inode dirty. - We could still use this (obviously except for - * NInoMstProtected() attributes, where we will need to duplicate the core code - * because we need our own async_io completion handler) but we could just do - * the i_size update in prepare write, when we resize the attribute. Then - * we would avoid the i_size update and mark_inode_dirty() happening here. + * updates i_size if the end of io is beyond i_size. In that case, it also + * marks the inode dirty. * - * Can't use generic_commit_write() due to ntfs specialities but can look at + * Cannot use generic_commit_write() due to ntfs specialities but can look at * it for implementation guidance. * * If things have gone as outlined in ntfs_prepare_write(), then we do not * need to do any page content modifications here at all, except in the write * to resident attribute case, where we need to do the uptodate bringing here - * which we combine with the copying into the mft record which means we only - * need to map the mft record and find the attribute record in it only once. + * which we combine with the copying into the mft record which means we save + * one atomic kmap. */ static int ntfs_commit_write(struct file *file, struct page *page, unsigned from, unsigned to) { - s64 attr_pos; - struct inode *vi; - ntfs_inode *ni, *base_ni; + struct inode *vi = page->mapping->host; + ntfs_inode *base_ni, *ni = NTFS_I(vi); char *kaddr, *kattr; ntfs_attr_search_ctx *ctx; MFT_RECORD *m; - u32 attr_len, bytes; + ATTR_RECORD *a; + u32 attr_len; int err; - vi = page->mapping->host; - ni = NTFS_I(vi); - ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index " "0x%lx, from = %u, to = %u.", vi->i_ino, ni->type, page->index, from, to); - + /* If the attribute is not resident, deal with it elsewhere. */ if (NInoNonResident(ni)) { - /* - * Only unnamed $DATA attributes can be compressed, encrypted, - * and/or sparse. - */ + /* Only unnamed $DATA attributes can be compressed/encrypted. */ if (ni->type == AT_DATA && !ni->name_len) { - /* If file is encrypted, deny access, just like NT4. */ + /* Encrypted files need separate handling. */ if (NInoEncrypted(ni)) { - // Should never get here! - ntfs_debug("Denying write access to encrypted " - "file."); - return -EACCES; + // We never get here at present! + BUG(); } /* Compressed data streams are handled in compress.c. */ if (NInoCompressed(ni)) { - // TODO: Implement and replace this check with + // TODO: Implement this! // return ntfs_write_compressed_block(page); - // Should never get here! - ntfs_error(vi->i_sb, "Writing to compressed " - "files is not supported yet. " - "Sorry."); - return -EOPNOTSUPP; + // We never get here at present! + BUG(); } - // TODO: Implement and remove this check. - if (NInoSparse(ni)) { - // Should never get here! - ntfs_error(vi->i_sb, "Writing to sparse files " - "is not supported yet. Sorry."); - return -EOPNOTSUPP; - } - } - - // TODO: Implement and remove this check. - if (NInoMstProtected(ni)) { - // Should never get here! - ntfs_error(vi->i_sb, "Writing to MST protected " - "attributes is not supported yet. " - "Sorry."); - return -EOPNOTSUPP; } - /* Normal data stream. */ return ntfs_commit_nonresident_write(page, from, to); } - /* * Attribute is resident, implying it is not compressed, encrypted, or - * mst protected. + * sparse. */ - - /* Do we need to resize the attribute? */ - if (((s64)page->index << PAGE_CACHE_SHIFT) + to > vi->i_size) { - // TODO: Implement resize... - // pos = ((s64)page->index << PAGE_CACHE_SHIFT) + to; - // vi->i_size = pos; - // mark_inode_dirty(vi); - // Should never get here! - ntfs_error(vi->i_sb, "Writing beyond the existing file size is " - "not supported yet. Sorry."); - return -EOPNOTSUPP; - } - if (!NInoAttr(ni)) base_ni = ni; else base_ni = ni->ext.base_ntfs_ino; - /* Map, pin, and lock the mft record. */ m = map_mft_record(base_ni); if (IS_ERR(m)) { @@ -2002,61 +2052,36 @@ } err = ntfs_attr_lookup(ni->type, ni->name, ni->name_len, CASE_SENSITIVE, 0, NULL, 0, ctx); - if (unlikely(err)) - goto err_out; - - /* Starting position of the page within the attribute value. */ - attr_pos = page->index << PAGE_CACHE_SHIFT; - - /* The total length of the attribute value. */ - attr_len = le32_to_cpu(ctx->attr->data.resident.value_length); - - if (unlikely(vi->i_size != attr_len)) { - ntfs_error(vi->i_sb, "BUG()! i_size (0x%llx) doesn't match " - "attr_len (0x%x). Aborting write.", vi->i_size, - attr_len); - err = -EIO; + if (unlikely(err)) { + if (err == -ENOENT) + err = -EIO; goto err_out; } - if (unlikely(attr_pos >= attr_len)) { - ntfs_error(vi->i_sb, "BUG()! attr_pos (0x%llx) > attr_len " - "(0x%x). Aborting write.", - (unsigned long long)attr_pos, attr_len); - err = -EIO; - goto err_out; - } - - bytes = attr_len - attr_pos; - if (unlikely(bytes > PAGE_CACHE_SIZE)) - bytes = PAGE_CACHE_SIZE; - - /* - * Calculate the address of the attribute value corresponding to the - * beginning of the current data @page. - */ - kattr = (u8*)ctx->attr + le16_to_cpu( - ctx->attr->data.resident.value_offset) + attr_pos; - + a = ctx->attr; + /* The total length of the attribute value. */ + attr_len = le32_to_cpu(a->data.resident.value_length); + BUG_ON(from > attr_len); + kattr = (u8*)a + le16_to_cpu(a->data.resident.value_offset); kaddr = kmap_atomic(page, KM_USER0); - /* Copy the received data from the page to the mft record. */ memcpy(kattr + from, kaddr + from, to - from); - flush_dcache_mft_record_page(ctx->ntfs_ino); - + /* Update the attribute length if necessary. */ + if (to > attr_len) { + attr_len = to; + a->data.resident.value_length = cpu_to_le32(attr_len); + } + /* + * If the page is not uptodate, bring the out of bounds area(s) + * uptodate by copying data from the mft record to the page. + */ if (!PageUptodate(page)) { - /* - * Bring the out of bounds area(s) uptodate by copying data - * from the mft record to the page. - */ if (from > 0) memcpy(kaddr, kattr, from); - if (to < bytes) - memcpy(kaddr + to, kattr + to, bytes - to); - + if (to < attr_len) + memcpy(kaddr + to, kattr + to, attr_len - to); /* Zero the region outside the end of the attribute value. */ - if (likely(bytes < PAGE_CACHE_SIZE)) - memset(kaddr + bytes, 0, PAGE_CACHE_SIZE - bytes); - + if (attr_len < PAGE_CACHE_SIZE) + memset(kaddr + attr_len, 0, PAGE_CACHE_SIZE - attr_len); /* * The probability of not having done any of the above is * extremely small, so we just flush unconditionally. @@ -2065,10 +2090,14 @@ SetPageUptodate(page); } kunmap_atomic(kaddr, KM_USER0); - + /* Update i_size if necessary. */ + if (vi->i_size < attr_len) { + ni->allocated_size = ni->initialized_size = attr_len; + i_size_write(vi, attr_len); + } /* Mark the mft record dirty, so it gets written back. */ + flush_dcache_mft_record_page(ctx->ntfs_ino); mark_mft_record_dirty(ctx->ntfs_ino); - ntfs_attr_put_search_ctx(ctx); unmap_mft_record(base_ni); ntfs_debug("Done."); @@ -2083,17 +2112,18 @@ "later on by the VM."); /* * Put the page on mapping->dirty_pages, but leave its - * buffer's dirty state as-is. + * buffers' dirty state as-is. */ __set_page_dirty_nobuffers(page); err = 0; } else - ntfs_error(vi->i_sb, "Page is not uptodate. Written " - "data has been lost. )-:"); + ntfs_error(vi->i_sb, "Page is not uptodate. Written " + "data has been lost."); } else { - ntfs_error(vi->i_sb, "Resident attribute write failed with " - "error %i. Setting page error flag.", -err); - SetPageError(page); + ntfs_error(vi->i_sb, "Resident attribute commit write failed " + "with error %i.", err); + NVolSetErrors(ni->vol); + make_bad_inode(vi); } if (ctx) ntfs_attr_put_search_ctx(ctx); |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:56
|
This is patch 10/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/28 1.2026.1.35) NTFS: In fs/ntfs/aops.c::ntfs_writepage(), if t he page is fully outside i_size, i.e. race with truncate, invalidate the buffers on the page so that they become freeable and hence the page does not leak. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:52 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:52 +00:00 @@ -42,6 +42,9 @@ mount time as this cannot work with the current implementation. - Check for location of attribute name and improve error handling in general in fs/ntfs/inode.c::ntfs_read_locked_inode() and friends. + - In fs/ntfs/aops.c::ntfs_writepage(), if t he page is fully outside + i_size, i.e. race with truncate, invalidate the buffers on the page + so that they become freeable and hence the page does not leak. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:44:52 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:44:52 +00:00 @@ -1117,7 +1117,8 @@ * For resident attributes, OTOH, ntfs_writepage() writes the @page by copying * the data to the mft record (which at this stage is most likely in memory). * The mft record is then marked dirty and written out asynchronously via the - * vfs inode dirty code path. + * vfs inode dirty code path for the inode the mft record belongs to or via the + * vm page dirty code path for the page the mft record is in. * * Based on ntfs_readpage() and fs/buffer.c::block_write_full_page(). * @@ -1141,6 +1142,11 @@ /* Is the page fully outside i_size? (truncate in progress) */ if (unlikely(page->index >= (vi->i_size + PAGE_CACHE_SIZE - 1) >> PAGE_CACHE_SHIFT)) { + /* + * The page may have dirty, unmapped buffers. Make them + * freeable here, so the page does not leak. + */ + block_invalidatepage(page, 0); unlock_page(page); ntfs_debug("Write outside i_size - truncated?"); return 0; |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:51
|
This is patch 8/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/27 1.2026.1.31) NTFS: Ensure the mft record size does not exceed the PAGE_CACHE_SIZE at mount time as this cannot work with the current implementation. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:44 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:44 +00:00 @@ -38,6 +38,8 @@ - In fs/ntfs/aops.c::mark_ntfs_record_dirty(), take the mapping->private_lock around the dirtying of the buffer heads analagous to the way it is done in __set_page_dirty_buffers(). + - Ensure the mft record size does not exceed the PAGE_CACHE_SIZE at + mount time as this cannot work with the current implementation. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/super.c b/fs/ntfs/super.c --- a/fs/ntfs/super.c 2004-11-10 13:44:44 +00:00 +++ b/fs/ntfs/super.c 2004-11-10 13:44:44 +00:00 @@ -680,7 +680,7 @@ * @b: boot sector to parse * * Parse the ntfs boot sector @b and store all imporant information therein in - * the ntfs super block @vol. Return TRUE on success and FALSE on error. + * the ntfs super block @vol. Return TRUE on success and FALSE on error. */ static BOOL parse_ntfs_boot_sector(ntfs_volume *vol, const NTFS_BOOT_SECTOR *b) { @@ -713,12 +713,12 @@ vol->cluster_size_bits, vol->cluster_size_bits); if (vol->sector_size > vol->cluster_size) { ntfs_error(vol->sb, "Sector sizes above the cluster size are " - "not supported. Sorry."); + "not supported. Sorry."); return FALSE; } if (vol->sb->s_blocksize > vol->cluster_size) { ntfs_error(vol->sb, "Cluster sizes smaller than the device " - "sector size are not supported. Sorry."); + "sector size are not supported. Sorry."); return FALSE; } clusters_per_mft_record = b->clusters_per_mft_record; @@ -742,6 +742,18 @@ vol->mft_record_size_mask); ntfs_debug("vol->mft_record_size_bits = %i (0x%x)", vol->mft_record_size_bits, vol->mft_record_size_bits); + /* + * We cannot support mft record sizes above the PAGE_CACHE_SIZE since + * we store $MFT/$DATA, the table of mft records in the page cache. + */ + if (vol->mft_record_size > PAGE_CACHE_SIZE) { + ntfs_error(vol->sb, "Mft record size %i (0x%x) exceeds the " + "page cache size on your system %lu (0x%lx). " + "This is not supported. Sorry.", + vol->mft_record_size, vol->mft_record_size, + PAGE_CACHE_SIZE, PAGE_CACHE_SIZE); + return FALSE; + } clusters_per_index_record = b->clusters_per_index_record; ntfs_debug("clusters_per_index_record = %i (0x%x)", clusters_per_index_record, clusters_per_index_record); @@ -772,7 +784,7 @@ */ ll = sle64_to_cpu(b->number_of_sectors) >> sectors_per_cluster_bits; if ((u64)ll >= 1ULL << 32) { - ntfs_error(vol->sb, "Cannot handle 64-bit clusters. Sorry."); + ntfs_error(vol->sb, "Cannot handle 64-bit clusters. Sorry."); return FALSE; } vol->nr_clusters = ll; @@ -785,8 +797,8 @@ if (sizeof(unsigned long) < 8) { if ((ll << vol->cluster_size_bits) >= (1ULL << 41)) { ntfs_error(vol->sb, "Volume size (%lluTiB) is too " - "large for this architecture. Maximum " - "supported is 2TiB. Sorry.", + "large for this architecture. " + "Maximum supported is 2TiB. Sorry.", (unsigned long long)ll >> (40 - vol->cluster_size_bits)); return FALSE; @@ -794,14 +806,14 @@ } ll = sle64_to_cpu(b->mft_lcn); if (ll >= vol->nr_clusters) { - ntfs_error(vol->sb, "MFT LCN is beyond end of volume. Weird."); + ntfs_error(vol->sb, "MFT LCN is beyond end of volume. Weird."); return FALSE; } vol->mft_lcn = ll; ntfs_debug("vol->mft_lcn = 0x%llx", (long long)vol->mft_lcn); ll = sle64_to_cpu(b->mftmirr_lcn); if (ll >= vol->nr_clusters) { - ntfs_error(vol->sb, "MFTMirr LCN is beyond end of volume. " + ntfs_error(vol->sb, "MFTMirr LCN is beyond end of volume. " "Weird."); return FALSE; } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:50
|
This is patch 4/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/25 1.2026.1.20) NTFS: - Change fs/ntfs/inode.c::ntfs_truncate() to return an error code instead of void and provide a helper ntfs_truncate_vfs() for the vfs ->truncate method. - Add a new ntfs inode flag NInoTruncateFailed() and modify fs/ntfs/inode.c::ntfs_truncate() to set and clear it appropriately. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:28 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:28 +00:00 @@ -24,6 +24,11 @@ 2.1.22-WIP - Improve error handling in fs/ntfs/inode.c::ntfs_truncate(). + - Change fs/ntfs/inode.c::ntfs_truncate() to return an error code + instead of void and provide a helper ntfs_truncate_vfs() for the + vfs ->truncate method. + - Add a new ntfs inode flag NInoTruncateFailed() and modify + fs/ntfs/inode.c::ntfs_truncate() to set and clear it appropriately. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/file.c b/fs/ntfs/file.c --- a/fs/ntfs/file.c 2004-11-10 13:44:28 +00:00 +++ b/fs/ntfs/file.c 2004-11-10 13:44:28 +00:00 @@ -145,7 +145,7 @@ struct inode_operations ntfs_file_inode_ops = { #ifdef NTFS_RW - .truncate = ntfs_truncate, + .truncate = ntfs_truncate_vfs, .setattr = ntfs_setattr, #endif /* NTFS_RW */ }; diff -Nru a/fs/ntfs/inode.c b/fs/ntfs/inode.c --- a/fs/ntfs/inode.c 2004-11-10 13:44:28 +00:00 +++ b/fs/ntfs/inode.c 2004-11-10 13:44:28 +00:00 @@ -2263,13 +2263,15 @@ * This implies for us that @vi is a file inode rather than a directory, index, * or attribute inode as well as that @vi is a base inode. * + * Returns 0 on success or -errno on error. + * * Called with ->i_sem held. In all but one case ->i_alloc_sem is held for * writing. The only case where ->i_alloc_sem is not held is * mm/filemap.c::generic_file_buffered_write() where vmtruncate() is called * with the current i_size as the offset which means that it is a noop as far * as ntfs_truncate() is concerned. */ -void ntfs_truncate(struct inode *vi) +int ntfs_truncate(struct inode *vi) { ntfs_inode *ni = NTFS_I(vi); ntfs_volume *vol = ni->vol; @@ -2323,7 +2325,8 @@ ntfs_attr_put_search_ctx(ctx); unmap_mft_record(ni); ntfs_debug("Done."); - return; + NInoClearTruncateFailed(ni); + return 0; err_out: if (err != -ENOMEM) { NVolSetErrors(vol); @@ -2333,7 +2336,20 @@ ntfs_attr_put_search_ctx(ctx); if (m) unmap_mft_record(ni); - return; + NInoSetTruncateFailed(ni); + return err; +} + +/** + * ntfs_truncate_vfs - wrapper for ntfs_truncate() that has no return value + * @vi: inode for which the i_size was changed + * + * Wrapper for ntfs_truncate() that has no return value. + * + * See ntfs_truncate() description above for details. + */ +void ntfs_truncate_vfs(struct inode *vi) { + ntfs_truncate(vi); } /** diff -Nru a/fs/ntfs/inode.h b/fs/ntfs/inode.h --- a/fs/ntfs/inode.h 2004-11-10 13:44:28 +00:00 +++ b/fs/ntfs/inode.h 2004-11-10 13:44:28 +00:00 @@ -165,6 +165,7 @@ NI_Sparse, /* 1: Unnamed data attr is sparse (f). 1: Create sparse files by default (d). 1: Attribute is sparse (a). */ + NI_TruncateFailed, /* 1: Last ntfs_truncate() call failed. */ } ntfs_inode_state_bits; /* @@ -216,6 +217,7 @@ NINO_FNS(Compressed) NINO_FNS(Encrypted) NINO_FNS(Sparse) +NINO_FNS(TruncateFailed) /* * The full structure containing a ntfs_inode and a vfs struct inode. Used for @@ -300,7 +302,8 @@ #ifdef NTFS_RW -extern void ntfs_truncate(struct inode *vi); +extern int ntfs_truncate(struct inode *vi); +extern void ntfs_truncate_vfs(struct inode *vi); extern int ntfs_setattr(struct dentry *dentry, struct iattr *attr); |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:49
|
This is patch 6/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/25 1.2026.1.22) NTFS: Add attribute definition handling helpers to fs/ntfs/attrib.[hc]: ntfs_attr_size_bounds_check(), ntfs_attr_can_be_non_resident(), and ntfs_attr_can_be_resident(), which in turn use the new private helper ntfs_attr_find_in_attrdef(). Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:36 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:36 +00:00 @@ -31,6 +31,10 @@ fs/ntfs/inode.c::ntfs_truncate() to set and clear it appropriately. - Fix min_size and max_size definitions in ATTR_DEF structure in fs/ntfs/layout.h to be signed. + - Add attribute definition handling helpers to fs/ntfs/attrib.[hc]: + ntfs_attr_size_bounds_check(), ntfs_attr_can_be_non_resident(), and + ntfs_attr_can_be_resident(), which in turn use the new private helper + ntfs_attr_find_in_attrdef(). 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/attrib.c b/fs/ntfs/attrib.c --- a/fs/ntfs/attrib.c 2004-11-10 13:44:36 +00:00 +++ b/fs/ntfs/attrib.c 2004-11-10 13:44:36 +00:00 @@ -24,8 +24,10 @@ #include "attrib.h" #include "debug.h" +#include "layout.h" #include "mft.h" #include "ntfs.h" +#include "types.h" /** * ntfs_map_runlist - map (a part of) a runlist of an ntfs inode @@ -947,6 +949,133 @@ unmap_extent_mft_record(ctx->ntfs_ino); kmem_cache_free(ntfs_attr_ctx_cache, ctx); return; +} + +/** + * ntfs_attr_find_in_attrdef - find an attribute in the $AttrDef system file + * @vol: ntfs volume to which the attribute belongs + * @type: attribute type which to find + * + * Search for the attribute definition record corresponding to the attribute + * @type in the $AttrDef system file. + * + * Return the attribute type definition record if found and NULL if not found. + */ +static ATTR_DEF *ntfs_attr_find_in_attrdef(const ntfs_volume *vol, + const ATTR_TYPE type) +{ + ATTR_DEF *ad; + + BUG_ON(!vol->attrdef); + BUG_ON(!type); + for (ad = vol->attrdef; (u8*)ad - (u8*)vol->attrdef < + vol->attrdef_size && ad->type; ++ad) { + /* We have not found it yet, carry on searching. */ + if (likely(le32_to_cpu(ad->type) < le32_to_cpu(type))) + continue; + /* We found the attribute; return it. */ + if (likely(ad->type == type)) + return ad; + /* We have gone too far already. No point in continuing. */ + break; + } + /* Attribute not found. */ + ntfs_debug("Attribute type 0x%x not found in $AttrDef.", + le32_to_cpu(type)); + return NULL; +} + +/** + * ntfs_attr_size_bounds_check - check a size of an attribute type for validity + * @vol: ntfs volume to which the attribute belongs + * @type: attribute type which to check + * @size: size which to check + * + * Check whether the @size in bytes is valid for an attribute of @type on the + * ntfs volume @vol. This information is obtained from $AttrDef system file. + * + * Return 0 if valid, -ERANGE if not valid, or -ENOENT if the attribute is not + * listed in $AttrDef. + */ +int ntfs_attr_size_bounds_check(const ntfs_volume *vol, const ATTR_TYPE type, + const s64 size) +{ + ATTR_DEF *ad; + + BUG_ON(size < 0); + /* + * $ATTRIBUTE_LIST has a maximum size of 256kiB, but this is not + * listed in $AttrDef. + */ + if (unlikely(type == AT_ATTRIBUTE_LIST && size > 256 * 1024)) + return -ERANGE; + /* Get the $AttrDef entry for the attribute @type. */ + ad = ntfs_attr_find_in_attrdef(vol, type); + if (unlikely(!ad)) + return -ENOENT; + /* Do the bounds check. */ + if (((sle64_to_cpu(ad->min_size) > 0) && + size < sle64_to_cpu(ad->min_size)) || + ((sle64_to_cpu(ad->max_size) > 0) && size > + sle64_to_cpu(ad->max_size))) + return -ERANGE; + return 0; +} + +/** + * ntfs_attr_can_be_non_resident - check if an attribute can be non-resident + * @vol: ntfs volume to which the attribute belongs + * @type: attribute type which to check + * + * Check whether the attribute of @type on the ntfs volume @vol is allowed to + * be non-resident. This information is obtained from $AttrDef system file. + * + * Return 0 if the attribute is allowed to be non-resident, -EPERM if not, or + * -ENOENT if the attribute is not listed in $AttrDef. + */ +int ntfs_attr_can_be_non_resident(const ntfs_volume *vol, const ATTR_TYPE type) +{ + ATTR_DEF *ad; + + /* + * $DATA is always allowed to be non-resident even if $AttrDef does not + * specify this in the flags of the $DATA attribute definition record. + */ + if (type == AT_DATA) + return 0; + /* Find the attribute definition record in $AttrDef. */ + ad = ntfs_attr_find_in_attrdef(vol, type); + if (unlikely(!ad)) + return -ENOENT; + /* Check the flags and return the result. */ + if (ad->flags & CAN_BE_NON_RESIDENT) + return 0; + return -EPERM; +} + +/** + * ntfs_attr_can_be_resident - check if an attribute can be resident + * @vol: ntfs volume to which the attribute belongs + * @type: attribute type which to check + * + * Check whether the attribute of @type on the ntfs volume @vol is allowed to + * be resident. This information is derived from our ntfs knowledge and may + * not be completely accurate, especially when user defined attributes are + * present. Basically we allow everything to be resident except for index + * allocation and $EA attributes. + * + * Return 0 if the attribute is allowed to be non-resident and -EPERM if not. + * + * Warning: In the system file $MFT the attribute $Bitmap must be non-resident + * otherwise windows will not boot (blue screen of death)! We cannot + * check for this here as we do not know which inode's $Bitmap is + * being asked about so the caller needs to special case this. + */ +int ntfs_attr_can_be_resident(const ntfs_volume *vol, const ATTR_TYPE type) +{ + if (type != AT_INDEX_ALLOCATION && type != AT_EA) + return 0; + return -EPERM; } /** diff -Nru a/fs/ntfs/attrib.h b/fs/ntfs/attrib.h --- a/fs/ntfs/attrib.h 2004-11-10 13:44:36 +00:00 +++ b/fs/ntfs/attrib.h 2004-11-10 13:44:36 +00:00 @@ -29,6 +29,7 @@ #include "layout.h" #include "inode.h" #include "runlist.h" +#include "volume.h" /** * ntfs_attr_search_ctx - used in attribute search functions @@ -83,6 +84,13 @@ extern ntfs_attr_search_ctx *ntfs_attr_get_search_ctx(ntfs_inode *ni, MFT_RECORD *mrec); extern void ntfs_attr_put_search_ctx(ntfs_attr_search_ctx *ctx); + +extern int ntfs_attr_size_bounds_check(const ntfs_volume *vol, + const ATTR_TYPE type, const s64 size); +extern int ntfs_attr_can_be_non_resident(const ntfs_volume *vol, + const ATTR_TYPE type); +extern int ntfs_attr_can_be_resident(const ntfs_volume *vol, + const ATTR_TYPE type); extern int ntfs_attr_record_resize(MFT_RECORD *m, ATTR_RECORD *a, u32 new_size); |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:45
|
This is patch 7/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/25 1.2026.1.23) NTFS: In fs/ntfs/aops.c::mark_ntfs_record_dirty(), take the mapping->private_lock around the dirtying of the buffer heads analagous to the way it is done in __set_page_dirty_buffers(). Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:40 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:40 +00:00 @@ -35,6 +35,9 @@ ntfs_attr_size_bounds_check(), ntfs_attr_can_be_non_resident(), and ntfs_attr_can_be_resident(), which in turn use the new private helper ntfs_attr_find_in_attrdef(). + - In fs/ntfs/aops.c::mark_ntfs_record_dirty(), take the + mapping->private_lock around the dirtying of the buffer heads + analagous to the way it is done in __set_page_dirty_buffers(). 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/aops.c b/fs/ntfs/aops.c --- a/fs/ntfs/aops.c 2004-11-10 13:44:40 +00:00 +++ b/fs/ntfs/aops.c 2004-11-10 13:44:40 +00:00 @@ -2158,6 +2158,7 @@ } end = ofs + ni->itype.index.block_size; bh_size = ni->vol->sb->s_blocksize; + spin_lock(&page->mapping->private_lock); bh = head = page_buffers(page); do { bh_ofs = bh_offset(bh); @@ -2167,6 +2168,7 @@ break; set_buffer_dirty(bh); } while ((bh = bh->b_this_page) != head); + spin_unlock(&page->mapping->private_lock); __set_page_dirty_nobuffers(page); } |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:45
|
This is patch 3/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/25 1.2026.1.19) NTFS: Add fs/ntfs/aops.c to list of providers for ->b_end_io method in Documentation/filesystems/Locking. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/Documentation/filesystems/Locking b/Documentation/filesystems/Locking --- a/Documentation/filesystems/Locking 2004-11-10 13:44:24 +00:00 +++ b/Documentation/filesystems/Locking 2004-11-10 13:44:24 +00:00 @@ -317,8 +317,8 @@ locking rules: called from interrupts. In other words, extreme care is needed here. bh is locked, but that's all warranties we have here. Currently only RAID1, -highmem and fs/buffer.c are providing these. Block devices call this method -upon the IO completion. +highmem, fs/buffer.c, and fs/ntfs/aops.c are providing these. Block devices +call this method upon the IO completion. --------------------------- block_device_operations ----------------------- prototypes: |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:41
|
This is patch 5/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/25 1.2026.1.21) NTFS: Fix min_size and max_size definitions in ATTR_DEF structure in fs/ntfs/layout.h to be signed. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/fs/ntfs/ChangeLog b/fs/ntfs/ChangeLog --- a/fs/ntfs/ChangeLog 2004-11-10 13:44:32 +00:00 +++ b/fs/ntfs/ChangeLog 2004-11-10 13:44:32 +00:00 @@ -29,6 +29,8 @@ vfs ->truncate method. - Add a new ntfs inode flag NInoTruncateFailed() and modify fs/ntfs/inode.c::ntfs_truncate() to set and clear it appropriately. + - Fix min_size and max_size definitions in ATTR_DEF structure in + fs/ntfs/layout.h to be signed. 2.1.21 - Fix some races and bugs, rewrite mft write code, add mft allocator. diff -Nru a/fs/ntfs/layout.h b/fs/ntfs/layout.h --- a/fs/ntfs/layout.h 2004-11-10 13:44:32 +00:00 +++ b/fs/ntfs/layout.h 2004-11-10 13:44:32 +00:00 @@ -589,8 +589,8 @@ FIXME: What does it mean? (AIA) */ /* 88*/ COLLATION_RULE collation_rule; /* Default collation rule. */ /* 8c*/ ATTR_DEF_FLAGS flags; /* Flags describing the attribute. */ -/* 90*/ le64 min_size; /* Optional minimum attribute size. */ -/* 98*/ le64 max_size; /* Maximum size of attribute. */ +/* 90*/ sle64 min_size; /* Optional minimum attribute size. */ +/* 98*/ sle64 max_size; /* Maximum size of attribute. */ /* sizeof() = 0xa0 or 160 bytes */ } __attribute__ ((__packed__)) ATTR_DEF; |
From: Anton A. <ai...@ca...> - 2004-11-10 13:44:34
|
This is patch 1/26 in the series. It contains the following ChangeSet: <ai...@ca...> (04/10/20 1.2000.1.12) NTFS: Fix two typos in Documentation/filesystems/ntfs.txt. Thanks to Richard Russon for pointing them out. Signed-off-by: Anton Altaparmakov <ai...@ca...> Best regards, Anton -- Anton Altaparmakov <aia21 at cam.ac.uk> (replace at with @) Unix Support, Computing Service, University of Cambridge, CB2 3QH, UK Linux NTFS maintainer / IRC: #ntfs on irc.freenode.net WWW: http://linux-ntfs.sf.net/, http://www-stu.christs.cam.ac.uk/~aia21/ =================================================================== diff -Nru a/Documentation/filesystems/ntfs.txt b/Documentation/filesystems/ntfs.txt --- a/Documentation/filesystems/ntfs.txt 2004-11-10 13:44:16 +00:00 +++ b/Documentation/filesystems/ntfs.txt 2004-11-10 13:44:16 +00:00 @@ -258,10 +258,10 @@ $ ./ldminfo --dump /dev/hda This would dump the LDM database found on /dev/hda which describes all of your -dinamic disks and all the volumes on them. At the bottom you will see the +dynamic disks and all the volumes on them. At the bottom you will see the VOLUME DEFINITIONS section which is all you really need. You may need to look further above to determine which of the disks in the volume definitions is -which device in Linux. Hint: Run ldminfo on each of your dinamic disks and +which device in Linux. Hint: Run ldminfo on each of your dynamic disks and look at the Disk Id close to the top of the output for each (the PRIVATE HEADER section). You can then find these Disk Ids in the VBLK DATABASE section in the <Disk> components where you will get the LDM Name for the disk that is found in |