Changes for 4.13:
- Avoid quotacheck deadlocks - Fix transaction overflows when bunmapping fragmented files - Refactor directory readahead - Allow admin to configure if ASSERT is fatal - Improve transaction usage detail logging during overflows - Minor cleanups - Don't leak log items when the log shuts down - Remove double-underscore typedefs - Various preparation for online scrubbing - Introduce new error injection configuration sysfs knobs - Refactor dq_get_next to use extent map directly - Fix problems with iterating the page cache for unwritten data - Implement SEEK_{HOLE,DATA} via iomap - Refactor XFS to use iomap SEEK_HOLE and SEEK_DATA - Don't use MAXPATHLEN to check on-disk symlink target lengths -----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQIcBAABCgAGBQJZYDw4AAoJEPh/dxk0SrTr2IMP/3JLeygIDtKBBVRPvlCmEXQC j8w1C/ntn46zZKQ8l14fAFV4HV2d+KJWf8+yDuPuGdMXJfPeKZf95otYhnSx/9Th MvCH7Nzg63yjEGqXpBkfIVr/GT0KTx28lxiqNViChr7XiXWookgf3SSLINO+vU4J L2jgLqieJfijiHTBs4qGCQPDwSXVoSOi5XCCQWDYQrXz6DI5UEJc70U53WkH4tRu RctOgp1lralwEO0PhfomD3m/Gk94taE/4ZpX/j/5Y4tvH/yh5aY3/KTCLm6+mYT3 rgMpmg5hmm+UiCTNoTnQ5RxzGZWCfI1I9FZ3HqDsbhmFtaWh32ti0dEEDYsF8Opj ARnTty3cRx41LH9dULrVWdwW105AHgwEz8/OZlG0JOca9qzj9GKERMg/hpHINAKN TrBlkweg86LWZDy23udZJ/v35svNqSFsqL1yV8j5dXyBi+Yi2CGfU27zbBwnj4Jk 047l+OuRbBnEOUULqJTEVBY3euoclwl/yQrW2m409s7vPGkGQBLuFCsDKQdnvJ/A D7frZqH8XypwnhFOkKybUnBkn4P7vZ2sEuCIZMsrH5k/ys8XyEkaBaOurjvMBOKA vLIMD6RXDWrFbOoovfK/stEM6/UFoQkgMhBe7vB9EXk1AjM8NYyWZgp5BkHtytC7 qa6GRjtGefhc67hbwXJd =/GZI -----END PGP SIGNATURE----- Merge tag 'xfs-4.13-merge-5' of git://git.kernel.org/pub/scm/fs/xfs/xfs-linux Pull XFS updates from Darrick Wong: "Here are some changes for you for 4.13. For the most part it's fixes for bugs and deadlock problems, and preparation for online fsck in some future merge window. - Avoid quotacheck deadlocks - Fix transaction overflows when bunmapping fragmented files - Refactor directory readahead - Allow admin to configure if ASSERT is fatal - Improve transaction usage detail logging during overflows - Minor cleanups - Don't leak log items when the log shuts down - Remove double-underscore typedefs - Various preparation for online scrubbing - Introduce new error injection configuration sysfs knobs - Refactor dq_get_next to use extent map directly - Fix problems with iterating the page cache for unwritten data - Implement SEEK_{HOLE,DATA} via iomap - Refactor XFS to use iomap SEEK_HOLE and SEEK_DATA - Don't use MAXPATHLEN to check on-disk symlink target lengths" * tag 'xfs-4.13-merge-5' of git://git.kernel.org/pub/scm/fs/xfs/xfs-linux: (48 commits) xfs: don't crash on unexpected holes in dir/attr btrees xfs: rename MAXPATHLEN to XFS_SYMLINK_MAXLEN xfs: fix contiguous dquot chunk iteration livelock xfs: Switch to iomap for SEEK_HOLE / SEEK_DATA vfs: Add iomap_seek_hole and iomap_seek_data helpers vfs: Add page_cache_seek_hole_data helper xfs: remove a whitespace-only line from xfs_fs_get_nextdqblk xfs: rewrite xfs_dq_get_next_id using xfs_iext_lookup_extent xfs: Check for m_errortag initialization in xfs_errortag_test xfs: grab dquots without taking the ilock xfs: fix semicolon.cocci warnings xfs: Don't clear SGID when inheriting ACLs xfs: free cowblocks and retry on buffered write ENOSPC xfs: replace log_badcrc_factor knob with error injection tag xfs: convert drop_writes to use the errortag mechanism xfs: remove unneeded parameter from XFS_TEST_ERROR xfs: expose errortag knobs via sysfs xfs: make errortag a per-mountpoint structure xfs: free uncommitted transactions during log recovery xfs: don't allow bmap on rt files ...
This commit is contained in:
commit
642338ba33
124
fs/buffer.c
124
fs/buffer.c
@ -3501,6 +3501,130 @@ int bh_submit_read(struct buffer_head *bh)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL(bh_submit_read);
|
EXPORT_SYMBOL(bh_submit_read);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Seek for SEEK_DATA / SEEK_HOLE within @page, starting at @lastoff.
|
||||||
|
*
|
||||||
|
* Returns the offset within the file on success, and -ENOENT otherwise.
|
||||||
|
*/
|
||||||
|
static loff_t
|
||||||
|
page_seek_hole_data(struct page *page, loff_t lastoff, int whence)
|
||||||
|
{
|
||||||
|
loff_t offset = page_offset(page);
|
||||||
|
struct buffer_head *bh, *head;
|
||||||
|
bool seek_data = whence == SEEK_DATA;
|
||||||
|
|
||||||
|
if (lastoff < offset)
|
||||||
|
lastoff = offset;
|
||||||
|
|
||||||
|
bh = head = page_buffers(page);
|
||||||
|
do {
|
||||||
|
offset += bh->b_size;
|
||||||
|
if (lastoff >= offset)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Unwritten extents that have data in the page cache covering
|
||||||
|
* them can be identified by the BH_Unwritten state flag.
|
||||||
|
* Pages with multiple buffers might have a mix of holes, data
|
||||||
|
* and unwritten extents - any buffer with valid data in it
|
||||||
|
* should have BH_Uptodate flag set on it.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if ((buffer_unwritten(bh) || buffer_uptodate(bh)) == seek_data)
|
||||||
|
return lastoff;
|
||||||
|
|
||||||
|
lastoff = offset;
|
||||||
|
} while ((bh = bh->b_this_page) != head);
|
||||||
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Seek for SEEK_DATA / SEEK_HOLE in the page cache.
|
||||||
|
*
|
||||||
|
* Within unwritten extents, the page cache determines which parts are holes
|
||||||
|
* and which are data: unwritten and uptodate buffer heads count as data;
|
||||||
|
* everything else counts as a hole.
|
||||||
|
*
|
||||||
|
* Returns the resulting offset on successs, and -ENOENT otherwise.
|
||||||
|
*/
|
||||||
|
loff_t
|
||||||
|
page_cache_seek_hole_data(struct inode *inode, loff_t offset, loff_t length,
|
||||||
|
int whence)
|
||||||
|
{
|
||||||
|
pgoff_t index = offset >> PAGE_SHIFT;
|
||||||
|
pgoff_t end = DIV_ROUND_UP(offset + length, PAGE_SIZE);
|
||||||
|
loff_t lastoff = offset;
|
||||||
|
struct pagevec pvec;
|
||||||
|
|
||||||
|
if (length <= 0)
|
||||||
|
return -ENOENT;
|
||||||
|
|
||||||
|
pagevec_init(&pvec, 0);
|
||||||
|
|
||||||
|
do {
|
||||||
|
unsigned want, nr_pages, i;
|
||||||
|
|
||||||
|
want = min_t(unsigned, end - index, PAGEVEC_SIZE);
|
||||||
|
nr_pages = pagevec_lookup(&pvec, inode->i_mapping, index, want);
|
||||||
|
if (nr_pages == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
for (i = 0; i < nr_pages; i++) {
|
||||||
|
struct page *page = pvec.pages[i];
|
||||||
|
|
||||||
|
/*
|
||||||
|
* At this point, the page may be truncated or
|
||||||
|
* invalidated (changing page->mapping to NULL), or
|
||||||
|
* even swizzled back from swapper_space to tmpfs file
|
||||||
|
* mapping. However, page->index will not change
|
||||||
|
* because we have a reference on the page.
|
||||||
|
*
|
||||||
|
* If current page offset is beyond where we've ended,
|
||||||
|
* we've found a hole.
|
||||||
|
*/
|
||||||
|
if (whence == SEEK_HOLE &&
|
||||||
|
lastoff < page_offset(page))
|
||||||
|
goto check_range;
|
||||||
|
|
||||||
|
/* Searching done if the page index is out of range. */
|
||||||
|
if (page->index >= end)
|
||||||
|
goto not_found;
|
||||||
|
|
||||||
|
lock_page(page);
|
||||||
|
if (likely(page->mapping == inode->i_mapping) &&
|
||||||
|
page_has_buffers(page)) {
|
||||||
|
lastoff = page_seek_hole_data(page, lastoff, whence);
|
||||||
|
if (lastoff >= 0) {
|
||||||
|
unlock_page(page);
|
||||||
|
goto check_range;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
unlock_page(page);
|
||||||
|
lastoff = page_offset(page) + PAGE_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Searching done if fewer pages returned than wanted. */
|
||||||
|
if (nr_pages < want)
|
||||||
|
break;
|
||||||
|
|
||||||
|
index = pvec.pages[i - 1]->index + 1;
|
||||||
|
pagevec_release(&pvec);
|
||||||
|
} while (index < end);
|
||||||
|
|
||||||
|
/* When no page at lastoff and we are not done, we found a hole. */
|
||||||
|
if (whence != SEEK_HOLE)
|
||||||
|
goto not_found;
|
||||||
|
|
||||||
|
check_range:
|
||||||
|
if (lastoff < offset + length)
|
||||||
|
goto out;
|
||||||
|
not_found:
|
||||||
|
lastoff = -ENOENT;
|
||||||
|
out:
|
||||||
|
pagevec_release(&pvec);
|
||||||
|
return lastoff;
|
||||||
|
}
|
||||||
|
|
||||||
void __init buffer_init(void)
|
void __init buffer_init(void)
|
||||||
{
|
{
|
||||||
unsigned long nrpages;
|
unsigned long nrpages;
|
||||||
|
94
fs/iomap.c
94
fs/iomap.c
@ -584,6 +584,100 @@ int iomap_fiemap(struct inode *inode, struct fiemap_extent_info *fi,
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(iomap_fiemap);
|
EXPORT_SYMBOL_GPL(iomap_fiemap);
|
||||||
|
|
||||||
|
static loff_t
|
||||||
|
iomap_seek_hole_actor(struct inode *inode, loff_t offset, loff_t length,
|
||||||
|
void *data, struct iomap *iomap)
|
||||||
|
{
|
||||||
|
switch (iomap->type) {
|
||||||
|
case IOMAP_UNWRITTEN:
|
||||||
|
offset = page_cache_seek_hole_data(inode, offset, length,
|
||||||
|
SEEK_HOLE);
|
||||||
|
if (offset < 0)
|
||||||
|
return length;
|
||||||
|
/* fall through */
|
||||||
|
case IOMAP_HOLE:
|
||||||
|
*(loff_t *)data = offset;
|
||||||
|
return 0;
|
||||||
|
default:
|
||||||
|
return length;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
loff_t
|
||||||
|
iomap_seek_hole(struct inode *inode, loff_t offset, const struct iomap_ops *ops)
|
||||||
|
{
|
||||||
|
loff_t size = i_size_read(inode);
|
||||||
|
loff_t length = size - offset;
|
||||||
|
loff_t ret;
|
||||||
|
|
||||||
|
/* Nothing to be found beyond the end of the file. */
|
||||||
|
if (offset >= size)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
|
while (length > 0) {
|
||||||
|
ret = iomap_apply(inode, offset, length, IOMAP_REPORT, ops,
|
||||||
|
&offset, iomap_seek_hole_actor);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
if (ret == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
offset += ret;
|
||||||
|
length -= ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return offset;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(iomap_seek_hole);
|
||||||
|
|
||||||
|
static loff_t
|
||||||
|
iomap_seek_data_actor(struct inode *inode, loff_t offset, loff_t length,
|
||||||
|
void *data, struct iomap *iomap)
|
||||||
|
{
|
||||||
|
switch (iomap->type) {
|
||||||
|
case IOMAP_HOLE:
|
||||||
|
return length;
|
||||||
|
case IOMAP_UNWRITTEN:
|
||||||
|
offset = page_cache_seek_hole_data(inode, offset, length,
|
||||||
|
SEEK_DATA);
|
||||||
|
if (offset < 0)
|
||||||
|
return length;
|
||||||
|
/*FALLTHRU*/
|
||||||
|
default:
|
||||||
|
*(loff_t *)data = offset;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
loff_t
|
||||||
|
iomap_seek_data(struct inode *inode, loff_t offset, const struct iomap_ops *ops)
|
||||||
|
{
|
||||||
|
loff_t size = i_size_read(inode);
|
||||||
|
loff_t length = size - offset;
|
||||||
|
loff_t ret;
|
||||||
|
|
||||||
|
/* Nothing to be found beyond the end of the file. */
|
||||||
|
if (offset >= size)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
|
while (length > 0) {
|
||||||
|
ret = iomap_apply(inode, offset, length, IOMAP_REPORT, ops,
|
||||||
|
&offset, iomap_seek_data_actor);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
if (ret == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
offset += ret;
|
||||||
|
length -= ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (length <= 0)
|
||||||
|
return -ENXIO;
|
||||||
|
return offset;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(iomap_seek_data);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Private flags for iomap_dio, must not overlap with the public ones in
|
* Private flags for iomap_dio, must not overlap with the public ones in
|
||||||
* iomap.h:
|
* iomap.h:
|
||||||
|
@ -96,3 +96,16 @@ config XFS_DEBUG
|
|||||||
not useful unless you are debugging a particular problem.
|
not useful unless you are debugging a particular problem.
|
||||||
|
|
||||||
Say N unless you are an XFS developer, or you play one on TV.
|
Say N unless you are an XFS developer, or you play one on TV.
|
||||||
|
|
||||||
|
config XFS_ASSERT_FATAL
|
||||||
|
bool "XFS fatal asserts"
|
||||||
|
default y
|
||||||
|
depends on XFS_FS && XFS_DEBUG
|
||||||
|
help
|
||||||
|
Set the default DEBUG mode ASSERT failure behavior.
|
||||||
|
|
||||||
|
Say Y here to cause DEBUG mode ASSERT failures to result in fatal
|
||||||
|
errors that BUG() the kernel by default. If you say N, ASSERT failures
|
||||||
|
result in warnings.
|
||||||
|
|
||||||
|
This behavior can be modified at runtime via sysfs.
|
||||||
|
@ -111,8 +111,7 @@ xfs_ag_resv_critical(
|
|||||||
|
|
||||||
/* Critically low if less than 10% or max btree height remains. */
|
/* Critically low if less than 10% or max btree height remains. */
|
||||||
return XFS_TEST_ERROR(avail < orig / 10 || avail < XFS_BTREE_MAXLEVELS,
|
return XFS_TEST_ERROR(avail < orig / 10 || avail < XFS_BTREE_MAXLEVELS,
|
||||||
pag->pag_mount, XFS_ERRTAG_AG_RESV_CRITICAL,
|
pag->pag_mount, XFS_ERRTAG_AG_RESV_CRITICAL);
|
||||||
XFS_RANDOM_AG_RESV_CRITICAL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -606,7 +606,7 @@ const struct xfs_buf_ops xfs_agfl_buf_ops = {
|
|||||||
/*
|
/*
|
||||||
* Read in the allocation group free block array.
|
* Read in the allocation group free block array.
|
||||||
*/
|
*/
|
||||||
STATIC int /* error */
|
int /* error */
|
||||||
xfs_alloc_read_agfl(
|
xfs_alloc_read_agfl(
|
||||||
xfs_mount_t *mp, /* mount point structure */
|
xfs_mount_t *mp, /* mount point structure */
|
||||||
xfs_trans_t *tp, /* transaction pointer */
|
xfs_trans_t *tp, /* transaction pointer */
|
||||||
@ -2454,8 +2454,7 @@ xfs_agf_read_verify(
|
|||||||
!xfs_buf_verify_cksum(bp, XFS_AGF_CRC_OFF))
|
!xfs_buf_verify_cksum(bp, XFS_AGF_CRC_OFF))
|
||||||
xfs_buf_ioerror(bp, -EFSBADCRC);
|
xfs_buf_ioerror(bp, -EFSBADCRC);
|
||||||
else if (XFS_TEST_ERROR(!xfs_agf_verify(mp, bp), mp,
|
else if (XFS_TEST_ERROR(!xfs_agf_verify(mp, bp), mp,
|
||||||
XFS_ERRTAG_ALLOC_READ_AGF,
|
XFS_ERRTAG_ALLOC_READ_AGF))
|
||||||
XFS_RANDOM_ALLOC_READ_AGF))
|
|
||||||
xfs_buf_ioerror(bp, -EFSCORRUPTED);
|
xfs_buf_ioerror(bp, -EFSCORRUPTED);
|
||||||
|
|
||||||
if (bp->b_error)
|
if (bp->b_error)
|
||||||
@ -2842,8 +2841,7 @@ xfs_free_extent(
|
|||||||
ASSERT(type != XFS_AG_RESV_AGFL);
|
ASSERT(type != XFS_AG_RESV_AGFL);
|
||||||
|
|
||||||
if (XFS_TEST_ERROR(false, mp,
|
if (XFS_TEST_ERROR(false, mp,
|
||||||
XFS_ERRTAG_FREE_EXTENT,
|
XFS_ERRTAG_FREE_EXTENT))
|
||||||
XFS_RANDOM_FREE_EXTENT))
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
error = xfs_free_extent_fix_freelist(tp, agno, &agbp);
|
error = xfs_free_extent_fix_freelist(tp, agno, &agbp);
|
||||||
|
@ -213,6 +213,8 @@ xfs_alloc_get_rec(
|
|||||||
|
|
||||||
int xfs_read_agf(struct xfs_mount *mp, struct xfs_trans *tp,
|
int xfs_read_agf(struct xfs_mount *mp, struct xfs_trans *tp,
|
||||||
xfs_agnumber_t agno, int flags, struct xfs_buf **bpp);
|
xfs_agnumber_t agno, int flags, struct xfs_buf **bpp);
|
||||||
|
int xfs_alloc_read_agfl(struct xfs_mount *mp, struct xfs_trans *tp,
|
||||||
|
xfs_agnumber_t agno, struct xfs_buf **bpp);
|
||||||
int xfs_alloc_fix_freelist(struct xfs_alloc_arg *args, int flags);
|
int xfs_alloc_fix_freelist(struct xfs_alloc_arg *args, int flags);
|
||||||
int xfs_free_extent_fix_freelist(struct xfs_trans *tp, xfs_agnumber_t agno,
|
int xfs_free_extent_fix_freelist(struct xfs_trans *tp, xfs_agnumber_t agno,
|
||||||
struct xfs_buf **agbp);
|
struct xfs_buf **agbp);
|
||||||
|
@ -253,7 +253,7 @@ xfs_allocbt_init_ptr_from_cur(
|
|||||||
ptr->s = agf->agf_roots[cur->bc_btnum];
|
ptr->s = agf->agf_roots[cur->bc_btnum];
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_bnobt_key_diff(
|
xfs_bnobt_key_diff(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key)
|
union xfs_btree_key *key)
|
||||||
@ -261,42 +261,42 @@ xfs_bnobt_key_diff(
|
|||||||
xfs_alloc_rec_incore_t *rec = &cur->bc_rec.a;
|
xfs_alloc_rec_incore_t *rec = &cur->bc_rec.a;
|
||||||
xfs_alloc_key_t *kp = &key->alloc;
|
xfs_alloc_key_t *kp = &key->alloc;
|
||||||
|
|
||||||
return (__int64_t)be32_to_cpu(kp->ar_startblock) - rec->ar_startblock;
|
return (int64_t)be32_to_cpu(kp->ar_startblock) - rec->ar_startblock;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_cntbt_key_diff(
|
xfs_cntbt_key_diff(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key)
|
union xfs_btree_key *key)
|
||||||
{
|
{
|
||||||
xfs_alloc_rec_incore_t *rec = &cur->bc_rec.a;
|
xfs_alloc_rec_incore_t *rec = &cur->bc_rec.a;
|
||||||
xfs_alloc_key_t *kp = &key->alloc;
|
xfs_alloc_key_t *kp = &key->alloc;
|
||||||
__int64_t diff;
|
int64_t diff;
|
||||||
|
|
||||||
diff = (__int64_t)be32_to_cpu(kp->ar_blockcount) - rec->ar_blockcount;
|
diff = (int64_t)be32_to_cpu(kp->ar_blockcount) - rec->ar_blockcount;
|
||||||
if (diff)
|
if (diff)
|
||||||
return diff;
|
return diff;
|
||||||
|
|
||||||
return (__int64_t)be32_to_cpu(kp->ar_startblock) - rec->ar_startblock;
|
return (int64_t)be32_to_cpu(kp->ar_startblock) - rec->ar_startblock;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_bnobt_diff_two_keys(
|
xfs_bnobt_diff_two_keys(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *k1,
|
union xfs_btree_key *k1,
|
||||||
union xfs_btree_key *k2)
|
union xfs_btree_key *k2)
|
||||||
{
|
{
|
||||||
return (__int64_t)be32_to_cpu(k1->alloc.ar_startblock) -
|
return (int64_t)be32_to_cpu(k1->alloc.ar_startblock) -
|
||||||
be32_to_cpu(k2->alloc.ar_startblock);
|
be32_to_cpu(k2->alloc.ar_startblock);
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_cntbt_diff_two_keys(
|
xfs_cntbt_diff_two_keys(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *k1,
|
union xfs_btree_key *k1,
|
||||||
union xfs_btree_key *k2)
|
union xfs_btree_key *k2)
|
||||||
{
|
{
|
||||||
__int64_t diff;
|
int64_t diff;
|
||||||
|
|
||||||
diff = be32_to_cpu(k1->alloc.ar_blockcount) -
|
diff = be32_to_cpu(k1->alloc.ar_blockcount) -
|
||||||
be32_to_cpu(k2->alloc.ar_blockcount);
|
be32_to_cpu(k2->alloc.ar_blockcount);
|
||||||
@ -395,7 +395,6 @@ const struct xfs_buf_ops xfs_allocbt_buf_ops = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
STATIC int
|
STATIC int
|
||||||
xfs_bnobt_keys_inorder(
|
xfs_bnobt_keys_inorder(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
@ -442,7 +441,6 @@ xfs_cntbt_recs_inorder(
|
|||||||
be32_to_cpu(r1->alloc.ar_startblock) <
|
be32_to_cpu(r1->alloc.ar_startblock) <
|
||||||
be32_to_cpu(r2->alloc.ar_startblock));
|
be32_to_cpu(r2->alloc.ar_startblock));
|
||||||
}
|
}
|
||||||
#endif /* DEBUG */
|
|
||||||
|
|
||||||
static const struct xfs_btree_ops xfs_bnobt_ops = {
|
static const struct xfs_btree_ops xfs_bnobt_ops = {
|
||||||
.rec_len = sizeof(xfs_alloc_rec_t),
|
.rec_len = sizeof(xfs_alloc_rec_t),
|
||||||
@ -462,10 +460,8 @@ static const struct xfs_btree_ops xfs_bnobt_ops = {
|
|||||||
.key_diff = xfs_bnobt_key_diff,
|
.key_diff = xfs_bnobt_key_diff,
|
||||||
.buf_ops = &xfs_allocbt_buf_ops,
|
.buf_ops = &xfs_allocbt_buf_ops,
|
||||||
.diff_two_keys = xfs_bnobt_diff_two_keys,
|
.diff_two_keys = xfs_bnobt_diff_two_keys,
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
.keys_inorder = xfs_bnobt_keys_inorder,
|
.keys_inorder = xfs_bnobt_keys_inorder,
|
||||||
.recs_inorder = xfs_bnobt_recs_inorder,
|
.recs_inorder = xfs_bnobt_recs_inorder,
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct xfs_btree_ops xfs_cntbt_ops = {
|
static const struct xfs_btree_ops xfs_cntbt_ops = {
|
||||||
@ -486,10 +482,8 @@ static const struct xfs_btree_ops xfs_cntbt_ops = {
|
|||||||
.key_diff = xfs_cntbt_key_diff,
|
.key_diff = xfs_cntbt_key_diff,
|
||||||
.buf_ops = &xfs_allocbt_buf_ops,
|
.buf_ops = &xfs_allocbt_buf_ops,
|
||||||
.diff_two_keys = xfs_cntbt_diff_two_keys,
|
.diff_two_keys = xfs_cntbt_diff_two_keys,
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
.keys_inorder = xfs_cntbt_keys_inorder,
|
.keys_inorder = xfs_cntbt_keys_inorder,
|
||||||
.recs_inorder = xfs_cntbt_recs_inorder,
|
.recs_inorder = xfs_cntbt_recs_inorder,
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -114,6 +114,23 @@ xfs_inode_hasattr(
|
|||||||
* Overall external interface routines.
|
* Overall external interface routines.
|
||||||
*========================================================================*/
|
*========================================================================*/
|
||||||
|
|
||||||
|
/* Retrieve an extended attribute and its value. Must have iolock. */
|
||||||
|
int
|
||||||
|
xfs_attr_get_ilocked(
|
||||||
|
struct xfs_inode *ip,
|
||||||
|
struct xfs_da_args *args)
|
||||||
|
{
|
||||||
|
if (!xfs_inode_hasattr(ip))
|
||||||
|
return -ENOATTR;
|
||||||
|
else if (ip->i_d.di_aformat == XFS_DINODE_FMT_LOCAL)
|
||||||
|
return xfs_attr_shortform_getvalue(args);
|
||||||
|
else if (xfs_bmap_one_block(ip, XFS_ATTR_FORK))
|
||||||
|
return xfs_attr_leaf_get(args);
|
||||||
|
else
|
||||||
|
return xfs_attr_node_get(args);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Retrieve an extended attribute by name, and its value. */
|
||||||
int
|
int
|
||||||
xfs_attr_get(
|
xfs_attr_get(
|
||||||
struct xfs_inode *ip,
|
struct xfs_inode *ip,
|
||||||
@ -141,14 +158,7 @@ xfs_attr_get(
|
|||||||
args.op_flags = XFS_DA_OP_OKNOENT;
|
args.op_flags = XFS_DA_OP_OKNOENT;
|
||||||
|
|
||||||
lock_mode = xfs_ilock_attr_map_shared(ip);
|
lock_mode = xfs_ilock_attr_map_shared(ip);
|
||||||
if (!xfs_inode_hasattr(ip))
|
error = xfs_attr_get_ilocked(ip, &args);
|
||||||
error = -ENOATTR;
|
|
||||||
else if (ip->i_d.di_aformat == XFS_DINODE_FMT_LOCAL)
|
|
||||||
error = xfs_attr_shortform_getvalue(&args);
|
|
||||||
else if (xfs_bmap_one_block(ip, XFS_ATTR_FORK))
|
|
||||||
error = xfs_attr_leaf_get(&args);
|
|
||||||
else
|
|
||||||
error = xfs_attr_node_get(&args);
|
|
||||||
xfs_iunlock(ip, lock_mode);
|
xfs_iunlock(ip, lock_mode);
|
||||||
|
|
||||||
*valuelenp = args.valuelen;
|
*valuelenp = args.valuelen;
|
||||||
|
@ -351,7 +351,7 @@ xfs_attr3_leaf_read(
|
|||||||
|
|
||||||
err = xfs_da_read_buf(tp, dp, bno, mappedbno, bpp,
|
err = xfs_da_read_buf(tp, dp, bno, mappedbno, bpp,
|
||||||
XFS_ATTR_FORK, &xfs_attr3_leaf_buf_ops);
|
XFS_ATTR_FORK, &xfs_attr3_leaf_buf_ops);
|
||||||
if (!err && tp)
|
if (!err && tp && *bpp)
|
||||||
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_ATTR_LEAF_BUF);
|
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_ATTR_LEAF_BUF);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
@ -253,7 +253,7 @@ xfs_attr_rmtval_copyout(
|
|||||||
xfs_ino_t ino,
|
xfs_ino_t ino,
|
||||||
int *offset,
|
int *offset,
|
||||||
int *valuelen,
|
int *valuelen,
|
||||||
__uint8_t **dst)
|
uint8_t **dst)
|
||||||
{
|
{
|
||||||
char *src = bp->b_addr;
|
char *src = bp->b_addr;
|
||||||
xfs_daddr_t bno = bp->b_bn;
|
xfs_daddr_t bno = bp->b_bn;
|
||||||
@ -301,7 +301,7 @@ xfs_attr_rmtval_copyin(
|
|||||||
xfs_ino_t ino,
|
xfs_ino_t ino,
|
||||||
int *offset,
|
int *offset,
|
||||||
int *valuelen,
|
int *valuelen,
|
||||||
__uint8_t **src)
|
uint8_t **src)
|
||||||
{
|
{
|
||||||
char *dst = bp->b_addr;
|
char *dst = bp->b_addr;
|
||||||
xfs_daddr_t bno = bp->b_bn;
|
xfs_daddr_t bno = bp->b_bn;
|
||||||
@ -355,7 +355,7 @@ xfs_attr_rmtval_get(
|
|||||||
struct xfs_mount *mp = args->dp->i_mount;
|
struct xfs_mount *mp = args->dp->i_mount;
|
||||||
struct xfs_buf *bp;
|
struct xfs_buf *bp;
|
||||||
xfs_dablk_t lblkno = args->rmtblkno;
|
xfs_dablk_t lblkno = args->rmtblkno;
|
||||||
__uint8_t *dst = args->value;
|
uint8_t *dst = args->value;
|
||||||
int valuelen;
|
int valuelen;
|
||||||
int nmap;
|
int nmap;
|
||||||
int error;
|
int error;
|
||||||
@ -386,7 +386,8 @@ xfs_attr_rmtval_get(
|
|||||||
(map[i].br_startblock != HOLESTARTBLOCK));
|
(map[i].br_startblock != HOLESTARTBLOCK));
|
||||||
dblkno = XFS_FSB_TO_DADDR(mp, map[i].br_startblock);
|
dblkno = XFS_FSB_TO_DADDR(mp, map[i].br_startblock);
|
||||||
dblkcnt = XFS_FSB_TO_BB(mp, map[i].br_blockcount);
|
dblkcnt = XFS_FSB_TO_BB(mp, map[i].br_blockcount);
|
||||||
error = xfs_trans_read_buf(mp, NULL, mp->m_ddev_targp,
|
error = xfs_trans_read_buf(mp, args->trans,
|
||||||
|
mp->m_ddev_targp,
|
||||||
dblkno, dblkcnt, 0, &bp,
|
dblkno, dblkcnt, 0, &bp,
|
||||||
&xfs_attr3_rmt_buf_ops);
|
&xfs_attr3_rmt_buf_ops);
|
||||||
if (error)
|
if (error)
|
||||||
@ -395,7 +396,7 @@ xfs_attr_rmtval_get(
|
|||||||
error = xfs_attr_rmtval_copyout(mp, bp, args->dp->i_ino,
|
error = xfs_attr_rmtval_copyout(mp, bp, args->dp->i_ino,
|
||||||
&offset, &valuelen,
|
&offset, &valuelen,
|
||||||
&dst);
|
&dst);
|
||||||
xfs_buf_relse(bp);
|
xfs_trans_brelse(args->trans, bp);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
|
|
||||||
@ -421,7 +422,7 @@ xfs_attr_rmtval_set(
|
|||||||
struct xfs_bmbt_irec map;
|
struct xfs_bmbt_irec map;
|
||||||
xfs_dablk_t lblkno;
|
xfs_dablk_t lblkno;
|
||||||
xfs_fileoff_t lfileoff = 0;
|
xfs_fileoff_t lfileoff = 0;
|
||||||
__uint8_t *src = args->value;
|
uint8_t *src = args->value;
|
||||||
int blkcnt;
|
int blkcnt;
|
||||||
int valuelen;
|
int valuelen;
|
||||||
int nmap;
|
int nmap;
|
||||||
|
@ -31,10 +31,10 @@ typedef struct xfs_attr_sf_entry xfs_attr_sf_entry_t;
|
|||||||
* We generate this then sort it, attr_list() must return things in hash-order.
|
* We generate this then sort it, attr_list() must return things in hash-order.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_attr_sf_sort {
|
typedef struct xfs_attr_sf_sort {
|
||||||
__uint8_t entno; /* entry number in original list */
|
uint8_t entno; /* entry number in original list */
|
||||||
__uint8_t namelen; /* length of name value (no null) */
|
uint8_t namelen; /* length of name value (no null) */
|
||||||
__uint8_t valuelen; /* length of value */
|
uint8_t valuelen; /* length of value */
|
||||||
__uint8_t flags; /* flags bits (see xfs_attr_leaf.h) */
|
uint8_t flags; /* flags bits (see xfs_attr_leaf.h) */
|
||||||
xfs_dahash_t hash; /* this entry's hash value */
|
xfs_dahash_t hash; /* this entry's hash value */
|
||||||
unsigned char *name; /* name value, pointer into buffer */
|
unsigned char *name; /* name value, pointer into buffer */
|
||||||
} xfs_attr_sf_sort_t;
|
} xfs_attr_sf_sort_t;
|
||||||
@ -42,7 +42,7 @@ typedef struct xfs_attr_sf_sort {
|
|||||||
#define XFS_ATTR_SF_ENTSIZE_BYNAME(nlen,vlen) /* space name/value uses */ \
|
#define XFS_ATTR_SF_ENTSIZE_BYNAME(nlen,vlen) /* space name/value uses */ \
|
||||||
(((int)sizeof(xfs_attr_sf_entry_t)-1 + (nlen)+(vlen)))
|
(((int)sizeof(xfs_attr_sf_entry_t)-1 + (nlen)+(vlen)))
|
||||||
#define XFS_ATTR_SF_ENTSIZE_MAX /* max space for name&value */ \
|
#define XFS_ATTR_SF_ENTSIZE_MAX /* max space for name&value */ \
|
||||||
((1 << (NBBY*(int)sizeof(__uint8_t))) - 1)
|
((1 << (NBBY*(int)sizeof(uint8_t))) - 1)
|
||||||
#define XFS_ATTR_SF_ENTSIZE(sfep) /* space an entry uses */ \
|
#define XFS_ATTR_SF_ENTSIZE(sfep) /* space an entry uses */ \
|
||||||
((int)sizeof(xfs_attr_sf_entry_t)-1 + (sfep)->namelen+(sfep)->valuelen)
|
((int)sizeof(xfs_attr_sf_entry_t)-1 + (sfep)->namelen+(sfep)->valuelen)
|
||||||
#define XFS_ATTR_SF_NEXTENTRY(sfep) /* next entry in struct */ \
|
#define XFS_ATTR_SF_NEXTENTRY(sfep) /* next entry in struct */ \
|
||||||
|
@ -25,47 +25,47 @@
|
|||||||
/*
|
/*
|
||||||
* masks with n high/low bits set, 64-bit values
|
* masks with n high/low bits set, 64-bit values
|
||||||
*/
|
*/
|
||||||
static inline __uint64_t xfs_mask64hi(int n)
|
static inline uint64_t xfs_mask64hi(int n)
|
||||||
{
|
{
|
||||||
return (__uint64_t)-1 << (64 - (n));
|
return (uint64_t)-1 << (64 - (n));
|
||||||
}
|
}
|
||||||
static inline __uint32_t xfs_mask32lo(int n)
|
static inline uint32_t xfs_mask32lo(int n)
|
||||||
{
|
{
|
||||||
return ((__uint32_t)1 << (n)) - 1;
|
return ((uint32_t)1 << (n)) - 1;
|
||||||
}
|
}
|
||||||
static inline __uint64_t xfs_mask64lo(int n)
|
static inline uint64_t xfs_mask64lo(int n)
|
||||||
{
|
{
|
||||||
return ((__uint64_t)1 << (n)) - 1;
|
return ((uint64_t)1 << (n)) - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get high bit set out of 32-bit argument, -1 if none set */
|
/* Get high bit set out of 32-bit argument, -1 if none set */
|
||||||
static inline int xfs_highbit32(__uint32_t v)
|
static inline int xfs_highbit32(uint32_t v)
|
||||||
{
|
{
|
||||||
return fls(v) - 1;
|
return fls(v) - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get high bit set out of 64-bit argument, -1 if none set */
|
/* Get high bit set out of 64-bit argument, -1 if none set */
|
||||||
static inline int xfs_highbit64(__uint64_t v)
|
static inline int xfs_highbit64(uint64_t v)
|
||||||
{
|
{
|
||||||
return fls64(v) - 1;
|
return fls64(v) - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get low bit set out of 32-bit argument, -1 if none set */
|
/* Get low bit set out of 32-bit argument, -1 if none set */
|
||||||
static inline int xfs_lowbit32(__uint32_t v)
|
static inline int xfs_lowbit32(uint32_t v)
|
||||||
{
|
{
|
||||||
return ffs(v) - 1;
|
return ffs(v) - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get low bit set out of 64-bit argument, -1 if none set */
|
/* Get low bit set out of 64-bit argument, -1 if none set */
|
||||||
static inline int xfs_lowbit64(__uint64_t v)
|
static inline int xfs_lowbit64(uint64_t v)
|
||||||
{
|
{
|
||||||
__uint32_t w = (__uint32_t)v;
|
uint32_t w = (uint32_t)v;
|
||||||
int n = 0;
|
int n = 0;
|
||||||
|
|
||||||
if (w) { /* lower bits */
|
if (w) { /* lower bits */
|
||||||
n = ffs(w);
|
n = ffs(w);
|
||||||
} else { /* upper bits */
|
} else { /* upper bits */
|
||||||
w = (__uint32_t)(v >> 32);
|
w = (uint32_t)(v >> 32);
|
||||||
if (w) {
|
if (w) {
|
||||||
n = ffs(w);
|
n = ffs(w);
|
||||||
if (n)
|
if (n)
|
||||||
|
@ -3992,7 +3992,7 @@ xfs_bmapi_read(
|
|||||||
if (unlikely(XFS_TEST_ERROR(
|
if (unlikely(XFS_TEST_ERROR(
|
||||||
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
||||||
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
||||||
mp, XFS_ERRTAG_BMAPIFORMAT, XFS_RANDOM_BMAPIFORMAT))) {
|
mp, XFS_ERRTAG_BMAPIFORMAT))) {
|
||||||
XFS_ERROR_REPORT("xfs_bmapi_read", XFS_ERRLEVEL_LOW, mp);
|
XFS_ERROR_REPORT("xfs_bmapi_read", XFS_ERRLEVEL_LOW, mp);
|
||||||
return -EFSCORRUPTED;
|
return -EFSCORRUPTED;
|
||||||
}
|
}
|
||||||
@ -4473,7 +4473,7 @@ xfs_bmapi_write(
|
|||||||
if (unlikely(XFS_TEST_ERROR(
|
if (unlikely(XFS_TEST_ERROR(
|
||||||
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
||||||
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
||||||
mp, XFS_ERRTAG_BMAPIFORMAT, XFS_RANDOM_BMAPIFORMAT))) {
|
mp, XFS_ERRTAG_BMAPIFORMAT))) {
|
||||||
XFS_ERROR_REPORT("xfs_bmapi_write", XFS_ERRLEVEL_LOW, mp);
|
XFS_ERROR_REPORT("xfs_bmapi_write", XFS_ERRLEVEL_LOW, mp);
|
||||||
return -EFSCORRUPTED;
|
return -EFSCORRUPTED;
|
||||||
}
|
}
|
||||||
@ -4694,7 +4694,7 @@ xfs_bmapi_remap(
|
|||||||
if (unlikely(XFS_TEST_ERROR(
|
if (unlikely(XFS_TEST_ERROR(
|
||||||
(XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_EXTENTS &&
|
(XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_EXTENTS &&
|
||||||
XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_BTREE),
|
XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_BTREE),
|
||||||
mp, XFS_ERRTAG_BMAPIFORMAT, XFS_RANDOM_BMAPIFORMAT))) {
|
mp, XFS_ERRTAG_BMAPIFORMAT))) {
|
||||||
XFS_ERROR_REPORT("xfs_bmapi_remap", XFS_ERRLEVEL_LOW, mp);
|
XFS_ERROR_REPORT("xfs_bmapi_remap", XFS_ERRLEVEL_LOW, mp);
|
||||||
return -EFSCORRUPTED;
|
return -EFSCORRUPTED;
|
||||||
}
|
}
|
||||||
@ -5434,6 +5434,7 @@ __xfs_bunmapi(
|
|||||||
int whichfork; /* data or attribute fork */
|
int whichfork; /* data or attribute fork */
|
||||||
xfs_fsblock_t sum;
|
xfs_fsblock_t sum;
|
||||||
xfs_filblks_t len = *rlen; /* length to unmap in file */
|
xfs_filblks_t len = *rlen; /* length to unmap in file */
|
||||||
|
xfs_fileoff_t max_len;
|
||||||
|
|
||||||
trace_xfs_bunmap(ip, bno, len, flags, _RET_IP_);
|
trace_xfs_bunmap(ip, bno, len, flags, _RET_IP_);
|
||||||
|
|
||||||
@ -5455,6 +5456,16 @@ __xfs_bunmapi(
|
|||||||
ASSERT(len > 0);
|
ASSERT(len > 0);
|
||||||
ASSERT(nexts >= 0);
|
ASSERT(nexts >= 0);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Guesstimate how many blocks we can unmap without running the risk of
|
||||||
|
* blowing out the transaction with a mix of EFIs and reflink
|
||||||
|
* adjustments.
|
||||||
|
*/
|
||||||
|
if (xfs_is_reflink_inode(ip) && whichfork == XFS_DATA_FORK)
|
||||||
|
max_len = min(len, xfs_refcount_max_unmap(tp->t_log_res));
|
||||||
|
else
|
||||||
|
max_len = len;
|
||||||
|
|
||||||
if (!(ifp->if_flags & XFS_IFEXTENTS) &&
|
if (!(ifp->if_flags & XFS_IFEXTENTS) &&
|
||||||
(error = xfs_iread_extents(tp, ip, whichfork)))
|
(error = xfs_iread_extents(tp, ip, whichfork)))
|
||||||
return error;
|
return error;
|
||||||
@ -5499,7 +5510,7 @@ __xfs_bunmapi(
|
|||||||
|
|
||||||
extno = 0;
|
extno = 0;
|
||||||
while (bno != (xfs_fileoff_t)-1 && bno >= start && lastx >= 0 &&
|
while (bno != (xfs_fileoff_t)-1 && bno >= start && lastx >= 0 &&
|
||||||
(nexts == 0 || extno < nexts)) {
|
(nexts == 0 || extno < nexts) && max_len > 0) {
|
||||||
/*
|
/*
|
||||||
* Is the found extent after a hole in which bno lives?
|
* Is the found extent after a hole in which bno lives?
|
||||||
* Just back up to the previous extent, if so.
|
* Just back up to the previous extent, if so.
|
||||||
@ -5531,6 +5542,15 @@ __xfs_bunmapi(
|
|||||||
}
|
}
|
||||||
if (del.br_startoff + del.br_blockcount > bno + 1)
|
if (del.br_startoff + del.br_blockcount > bno + 1)
|
||||||
del.br_blockcount = bno + 1 - del.br_startoff;
|
del.br_blockcount = bno + 1 - del.br_startoff;
|
||||||
|
|
||||||
|
/* How much can we safely unmap? */
|
||||||
|
if (max_len < del.br_blockcount) {
|
||||||
|
del.br_startoff += del.br_blockcount - max_len;
|
||||||
|
if (!wasdel)
|
||||||
|
del.br_startblock += del.br_blockcount - max_len;
|
||||||
|
del.br_blockcount = max_len;
|
||||||
|
}
|
||||||
|
|
||||||
sum = del.br_startblock + del.br_blockcount;
|
sum = del.br_startblock + del.br_blockcount;
|
||||||
if (isrt &&
|
if (isrt &&
|
||||||
(mod = do_mod(sum, mp->m_sb.sb_rextsize))) {
|
(mod = do_mod(sum, mp->m_sb.sb_rextsize))) {
|
||||||
@ -5707,6 +5727,7 @@ __xfs_bunmapi(
|
|||||||
if (!isrt && wasdel)
|
if (!isrt && wasdel)
|
||||||
xfs_mod_fdblocks(mp, (int64_t)del.br_blockcount, false);
|
xfs_mod_fdblocks(mp, (int64_t)del.br_blockcount, false);
|
||||||
|
|
||||||
|
max_len -= del.br_blockcount;
|
||||||
bno = del.br_startoff - 1;
|
bno = del.br_startoff - 1;
|
||||||
nodelete:
|
nodelete:
|
||||||
/*
|
/*
|
||||||
@ -6077,7 +6098,7 @@ xfs_bmap_shift_extents(
|
|||||||
if (unlikely(XFS_TEST_ERROR(
|
if (unlikely(XFS_TEST_ERROR(
|
||||||
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
||||||
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
||||||
mp, XFS_ERRTAG_BMAPIFORMAT, XFS_RANDOM_BMAPIFORMAT))) {
|
mp, XFS_ERRTAG_BMAPIFORMAT))) {
|
||||||
XFS_ERROR_REPORT("xfs_bmap_shift_extents",
|
XFS_ERROR_REPORT("xfs_bmap_shift_extents",
|
||||||
XFS_ERRLEVEL_LOW, mp);
|
XFS_ERRLEVEL_LOW, mp);
|
||||||
return -EFSCORRUPTED;
|
return -EFSCORRUPTED;
|
||||||
@ -6229,7 +6250,7 @@ xfs_bmap_split_extent_at(
|
|||||||
if (unlikely(XFS_TEST_ERROR(
|
if (unlikely(XFS_TEST_ERROR(
|
||||||
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
(XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_EXTENTS &&
|
||||||
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
XFS_IFORK_FORMAT(ip, whichfork) != XFS_DINODE_FMT_BTREE),
|
||||||
mp, XFS_ERRTAG_BMAPIFORMAT, XFS_RANDOM_BMAPIFORMAT))) {
|
mp, XFS_ERRTAG_BMAPIFORMAT))) {
|
||||||
XFS_ERROR_REPORT("xfs_bmap_split_extent_at",
|
XFS_ERROR_REPORT("xfs_bmap_split_extent_at",
|
||||||
XFS_ERRLEVEL_LOW, mp);
|
XFS_ERRLEVEL_LOW, mp);
|
||||||
return -EFSCORRUPTED;
|
return -EFSCORRUPTED;
|
||||||
@ -6472,33 +6493,33 @@ xfs_bmap_finish_one(
|
|||||||
int whichfork,
|
int whichfork,
|
||||||
xfs_fileoff_t startoff,
|
xfs_fileoff_t startoff,
|
||||||
xfs_fsblock_t startblock,
|
xfs_fsblock_t startblock,
|
||||||
xfs_filblks_t blockcount,
|
xfs_filblks_t *blockcount,
|
||||||
xfs_exntst_t state)
|
xfs_exntst_t state)
|
||||||
{
|
{
|
||||||
int error = 0, done;
|
xfs_fsblock_t firstfsb;
|
||||||
|
int error = 0;
|
||||||
|
|
||||||
trace_xfs_bmap_deferred(tp->t_mountp,
|
trace_xfs_bmap_deferred(tp->t_mountp,
|
||||||
XFS_FSB_TO_AGNO(tp->t_mountp, startblock), type,
|
XFS_FSB_TO_AGNO(tp->t_mountp, startblock), type,
|
||||||
XFS_FSB_TO_AGBNO(tp->t_mountp, startblock),
|
XFS_FSB_TO_AGBNO(tp->t_mountp, startblock),
|
||||||
ip->i_ino, whichfork, startoff, blockcount, state);
|
ip->i_ino, whichfork, startoff, *blockcount, state);
|
||||||
|
|
||||||
if (WARN_ON_ONCE(whichfork != XFS_DATA_FORK))
|
if (WARN_ON_ONCE(whichfork != XFS_DATA_FORK))
|
||||||
return -EFSCORRUPTED;
|
return -EFSCORRUPTED;
|
||||||
|
|
||||||
if (XFS_TEST_ERROR(false, tp->t_mountp,
|
if (XFS_TEST_ERROR(false, tp->t_mountp,
|
||||||
XFS_ERRTAG_BMAP_FINISH_ONE,
|
XFS_ERRTAG_BMAP_FINISH_ONE))
|
||||||
XFS_RANDOM_BMAP_FINISH_ONE))
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case XFS_BMAP_MAP:
|
case XFS_BMAP_MAP:
|
||||||
error = xfs_bmapi_remap(tp, ip, startoff, blockcount,
|
error = xfs_bmapi_remap(tp, ip, startoff, *blockcount,
|
||||||
startblock, dfops);
|
startblock, dfops);
|
||||||
|
*blockcount = 0;
|
||||||
break;
|
break;
|
||||||
case XFS_BMAP_UNMAP:
|
case XFS_BMAP_UNMAP:
|
||||||
error = xfs_bunmapi(tp, ip, startoff, blockcount,
|
error = __xfs_bunmapi(tp, ip, startoff, blockcount,
|
||||||
XFS_BMAPI_REMAP, 1, &startblock, dfops, &done);
|
XFS_BMAPI_REMAP, 1, &firstfsb, dfops);
|
||||||
ASSERT(done);
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
ASSERT(0);
|
ASSERT(0);
|
||||||
|
@ -271,7 +271,7 @@ struct xfs_bmap_intent {
|
|||||||
int xfs_bmap_finish_one(struct xfs_trans *tp, struct xfs_defer_ops *dfops,
|
int xfs_bmap_finish_one(struct xfs_trans *tp, struct xfs_defer_ops *dfops,
|
||||||
struct xfs_inode *ip, enum xfs_bmap_intent_type type,
|
struct xfs_inode *ip, enum xfs_bmap_intent_type type,
|
||||||
int whichfork, xfs_fileoff_t startoff, xfs_fsblock_t startblock,
|
int whichfork, xfs_fileoff_t startoff, xfs_fsblock_t startblock,
|
||||||
xfs_filblks_t blockcount, xfs_exntst_t state);
|
xfs_filblks_t *blockcount, xfs_exntst_t state);
|
||||||
int xfs_bmap_map_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
int xfs_bmap_map_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
||||||
struct xfs_inode *ip, struct xfs_bmbt_irec *imap);
|
struct xfs_inode *ip, struct xfs_bmbt_irec *imap);
|
||||||
int xfs_bmap_unmap_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
int xfs_bmap_unmap_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
||||||
|
@ -94,8 +94,8 @@ xfs_bmdr_to_bmbt(
|
|||||||
*/
|
*/
|
||||||
STATIC void
|
STATIC void
|
||||||
__xfs_bmbt_get_all(
|
__xfs_bmbt_get_all(
|
||||||
__uint64_t l0,
|
uint64_t l0,
|
||||||
__uint64_t l1,
|
uint64_t l1,
|
||||||
xfs_bmbt_irec_t *s)
|
xfs_bmbt_irec_t *s)
|
||||||
{
|
{
|
||||||
int ext_flag;
|
int ext_flag;
|
||||||
@ -572,6 +572,16 @@ xfs_bmbt_init_key_from_rec(
|
|||||||
cpu_to_be64(xfs_bmbt_disk_get_startoff(&rec->bmbt));
|
cpu_to_be64(xfs_bmbt_disk_get_startoff(&rec->bmbt));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
STATIC void
|
||||||
|
xfs_bmbt_init_high_key_from_rec(
|
||||||
|
union xfs_btree_key *key,
|
||||||
|
union xfs_btree_rec *rec)
|
||||||
|
{
|
||||||
|
key->bmbt.br_startoff = cpu_to_be64(
|
||||||
|
xfs_bmbt_disk_get_startoff(&rec->bmbt) +
|
||||||
|
xfs_bmbt_disk_get_blockcount(&rec->bmbt) - 1);
|
||||||
|
}
|
||||||
|
|
||||||
STATIC void
|
STATIC void
|
||||||
xfs_bmbt_init_rec_from_cur(
|
xfs_bmbt_init_rec_from_cur(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
@ -588,15 +598,25 @@ xfs_bmbt_init_ptr_from_cur(
|
|||||||
ptr->l = 0;
|
ptr->l = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_bmbt_key_diff(
|
xfs_bmbt_key_diff(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key)
|
union xfs_btree_key *key)
|
||||||
{
|
{
|
||||||
return (__int64_t)be64_to_cpu(key->bmbt.br_startoff) -
|
return (int64_t)be64_to_cpu(key->bmbt.br_startoff) -
|
||||||
cur->bc_rec.b.br_startoff;
|
cur->bc_rec.b.br_startoff;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
STATIC int64_t
|
||||||
|
xfs_bmbt_diff_two_keys(
|
||||||
|
struct xfs_btree_cur *cur,
|
||||||
|
union xfs_btree_key *k1,
|
||||||
|
union xfs_btree_key *k2)
|
||||||
|
{
|
||||||
|
return (int64_t)be64_to_cpu(k1->bmbt.br_startoff) -
|
||||||
|
be64_to_cpu(k2->bmbt.br_startoff);
|
||||||
|
}
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
xfs_bmbt_verify(
|
xfs_bmbt_verify(
|
||||||
struct xfs_buf *bp)
|
struct xfs_buf *bp)
|
||||||
@ -687,7 +707,6 @@ const struct xfs_buf_ops xfs_bmbt_buf_ops = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
STATIC int
|
STATIC int
|
||||||
xfs_bmbt_keys_inorder(
|
xfs_bmbt_keys_inorder(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
@ -708,7 +727,6 @@ xfs_bmbt_recs_inorder(
|
|||||||
xfs_bmbt_disk_get_blockcount(&r1->bmbt) <=
|
xfs_bmbt_disk_get_blockcount(&r1->bmbt) <=
|
||||||
xfs_bmbt_disk_get_startoff(&r2->bmbt);
|
xfs_bmbt_disk_get_startoff(&r2->bmbt);
|
||||||
}
|
}
|
||||||
#endif /* DEBUG */
|
|
||||||
|
|
||||||
static const struct xfs_btree_ops xfs_bmbt_ops = {
|
static const struct xfs_btree_ops xfs_bmbt_ops = {
|
||||||
.rec_len = sizeof(xfs_bmbt_rec_t),
|
.rec_len = sizeof(xfs_bmbt_rec_t),
|
||||||
@ -722,14 +740,14 @@ static const struct xfs_btree_ops xfs_bmbt_ops = {
|
|||||||
.get_minrecs = xfs_bmbt_get_minrecs,
|
.get_minrecs = xfs_bmbt_get_minrecs,
|
||||||
.get_dmaxrecs = xfs_bmbt_get_dmaxrecs,
|
.get_dmaxrecs = xfs_bmbt_get_dmaxrecs,
|
||||||
.init_key_from_rec = xfs_bmbt_init_key_from_rec,
|
.init_key_from_rec = xfs_bmbt_init_key_from_rec,
|
||||||
|
.init_high_key_from_rec = xfs_bmbt_init_high_key_from_rec,
|
||||||
.init_rec_from_cur = xfs_bmbt_init_rec_from_cur,
|
.init_rec_from_cur = xfs_bmbt_init_rec_from_cur,
|
||||||
.init_ptr_from_cur = xfs_bmbt_init_ptr_from_cur,
|
.init_ptr_from_cur = xfs_bmbt_init_ptr_from_cur,
|
||||||
.key_diff = xfs_bmbt_key_diff,
|
.key_diff = xfs_bmbt_key_diff,
|
||||||
|
.diff_two_keys = xfs_bmbt_diff_two_keys,
|
||||||
.buf_ops = &xfs_bmbt_buf_ops,
|
.buf_ops = &xfs_bmbt_buf_ops,
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
.keys_inorder = xfs_bmbt_keys_inorder,
|
.keys_inorder = xfs_bmbt_keys_inorder,
|
||||||
.recs_inorder = xfs_bmbt_recs_inorder,
|
.recs_inorder = xfs_bmbt_recs_inorder,
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -43,7 +43,7 @@ kmem_zone_t *xfs_btree_cur_zone;
|
|||||||
/*
|
/*
|
||||||
* Btree magic numbers.
|
* Btree magic numbers.
|
||||||
*/
|
*/
|
||||||
static const __uint32_t xfs_magics[2][XFS_BTNUM_MAX] = {
|
static const uint32_t xfs_magics[2][XFS_BTNUM_MAX] = {
|
||||||
{ XFS_ABTB_MAGIC, XFS_ABTC_MAGIC, 0, XFS_BMAP_MAGIC, XFS_IBT_MAGIC,
|
{ XFS_ABTB_MAGIC, XFS_ABTC_MAGIC, 0, XFS_BMAP_MAGIC, XFS_IBT_MAGIC,
|
||||||
XFS_FIBT_MAGIC, 0 },
|
XFS_FIBT_MAGIC, 0 },
|
||||||
{ XFS_ABTB_CRC_MAGIC, XFS_ABTC_CRC_MAGIC, XFS_RMAP_CRC_MAGIC,
|
{ XFS_ABTB_CRC_MAGIC, XFS_ABTC_CRC_MAGIC, XFS_RMAP_CRC_MAGIC,
|
||||||
@ -51,12 +51,12 @@ static const __uint32_t xfs_magics[2][XFS_BTNUM_MAX] = {
|
|||||||
XFS_REFC_CRC_MAGIC }
|
XFS_REFC_CRC_MAGIC }
|
||||||
};
|
};
|
||||||
|
|
||||||
__uint32_t
|
uint32_t
|
||||||
xfs_btree_magic(
|
xfs_btree_magic(
|
||||||
int crc,
|
int crc,
|
||||||
xfs_btnum_t btnum)
|
xfs_btnum_t btnum)
|
||||||
{
|
{
|
||||||
__uint32_t magic = xfs_magics[crc][btnum];
|
uint32_t magic = xfs_magics[crc][btnum];
|
||||||
|
|
||||||
/* Ensure we asked for crc for crc-only magics. */
|
/* Ensure we asked for crc for crc-only magics. */
|
||||||
ASSERT(magic != 0);
|
ASSERT(magic != 0);
|
||||||
@ -101,8 +101,7 @@ xfs_btree_check_lblock(
|
|||||||
be64_to_cpu(block->bb_u.l.bb_rightsib)));
|
be64_to_cpu(block->bb_u.l.bb_rightsib)));
|
||||||
|
|
||||||
if (unlikely(XFS_TEST_ERROR(!lblock_ok, mp,
|
if (unlikely(XFS_TEST_ERROR(!lblock_ok, mp,
|
||||||
XFS_ERRTAG_BTREE_CHECK_LBLOCK,
|
XFS_ERRTAG_BTREE_CHECK_LBLOCK))) {
|
||||||
XFS_RANDOM_BTREE_CHECK_LBLOCK))) {
|
|
||||||
if (bp)
|
if (bp)
|
||||||
trace_xfs_btree_corrupt(bp, _RET_IP_);
|
trace_xfs_btree_corrupt(bp, _RET_IP_);
|
||||||
XFS_ERROR_REPORT(__func__, XFS_ERRLEVEL_LOW, mp);
|
XFS_ERROR_REPORT(__func__, XFS_ERRLEVEL_LOW, mp);
|
||||||
@ -153,8 +152,7 @@ xfs_btree_check_sblock(
|
|||||||
block->bb_u.s.bb_rightsib;
|
block->bb_u.s.bb_rightsib;
|
||||||
|
|
||||||
if (unlikely(XFS_TEST_ERROR(!sblock_ok, mp,
|
if (unlikely(XFS_TEST_ERROR(!sblock_ok, mp,
|
||||||
XFS_ERRTAG_BTREE_CHECK_SBLOCK,
|
XFS_ERRTAG_BTREE_CHECK_SBLOCK))) {
|
||||||
XFS_RANDOM_BTREE_CHECK_SBLOCK))) {
|
|
||||||
if (bp)
|
if (bp)
|
||||||
trace_xfs_btree_corrupt(bp, _RET_IP_);
|
trace_xfs_btree_corrupt(bp, _RET_IP_);
|
||||||
XFS_ERROR_REPORT(__func__, XFS_ERRLEVEL_LOW, mp);
|
XFS_ERROR_REPORT(__func__, XFS_ERRLEVEL_LOW, mp);
|
||||||
@ -568,7 +566,7 @@ xfs_btree_ptr_offset(
|
|||||||
/*
|
/*
|
||||||
* Return a pointer to the n-th record in the btree block.
|
* Return a pointer to the n-th record in the btree block.
|
||||||
*/
|
*/
|
||||||
STATIC union xfs_btree_rec *
|
union xfs_btree_rec *
|
||||||
xfs_btree_rec_addr(
|
xfs_btree_rec_addr(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
int n,
|
int n,
|
||||||
@ -581,7 +579,7 @@ xfs_btree_rec_addr(
|
|||||||
/*
|
/*
|
||||||
* Return a pointer to the n-th key in the btree block.
|
* Return a pointer to the n-th key in the btree block.
|
||||||
*/
|
*/
|
||||||
STATIC union xfs_btree_key *
|
union xfs_btree_key *
|
||||||
xfs_btree_key_addr(
|
xfs_btree_key_addr(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
int n,
|
int n,
|
||||||
@ -594,7 +592,7 @@ xfs_btree_key_addr(
|
|||||||
/*
|
/*
|
||||||
* Return a pointer to the n-th high key in the btree block.
|
* Return a pointer to the n-th high key in the btree block.
|
||||||
*/
|
*/
|
||||||
STATIC union xfs_btree_key *
|
union xfs_btree_key *
|
||||||
xfs_btree_high_key_addr(
|
xfs_btree_high_key_addr(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
int n,
|
int n,
|
||||||
@ -607,7 +605,7 @@ xfs_btree_high_key_addr(
|
|||||||
/*
|
/*
|
||||||
* Return a pointer to the n-th block pointer in the btree block.
|
* Return a pointer to the n-th block pointer in the btree block.
|
||||||
*/
|
*/
|
||||||
STATIC union xfs_btree_ptr *
|
union xfs_btree_ptr *
|
||||||
xfs_btree_ptr_addr(
|
xfs_btree_ptr_addr(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
int n,
|
int n,
|
||||||
@ -641,7 +639,7 @@ xfs_btree_get_iroot(
|
|||||||
* Retrieve the block pointer from the cursor at the given level.
|
* Retrieve the block pointer from the cursor at the given level.
|
||||||
* This may be an inode btree root or from a buffer.
|
* This may be an inode btree root or from a buffer.
|
||||||
*/
|
*/
|
||||||
STATIC struct xfs_btree_block * /* generic btree block pointer */
|
struct xfs_btree_block * /* generic btree block pointer */
|
||||||
xfs_btree_get_block(
|
xfs_btree_get_block(
|
||||||
struct xfs_btree_cur *cur, /* btree cursor */
|
struct xfs_btree_cur *cur, /* btree cursor */
|
||||||
int level, /* level in btree */
|
int level, /* level in btree */
|
||||||
@ -778,14 +776,14 @@ xfs_btree_lastrec(
|
|||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
xfs_btree_offsets(
|
xfs_btree_offsets(
|
||||||
__int64_t fields, /* bitmask of fields */
|
int64_t fields, /* bitmask of fields */
|
||||||
const short *offsets, /* table of field offsets */
|
const short *offsets, /* table of field offsets */
|
||||||
int nbits, /* number of bits to inspect */
|
int nbits, /* number of bits to inspect */
|
||||||
int *first, /* output: first byte offset */
|
int *first, /* output: first byte offset */
|
||||||
int *last) /* output: last byte offset */
|
int *last) /* output: last byte offset */
|
||||||
{
|
{
|
||||||
int i; /* current bit number */
|
int i; /* current bit number */
|
||||||
__int64_t imask; /* mask for current bit number */
|
int64_t imask; /* mask for current bit number */
|
||||||
|
|
||||||
ASSERT(fields != 0);
|
ASSERT(fields != 0);
|
||||||
/*
|
/*
|
||||||
@ -1756,7 +1754,7 @@ error0:
|
|||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC int
|
int
|
||||||
xfs_btree_lookup_get_block(
|
xfs_btree_lookup_get_block(
|
||||||
struct xfs_btree_cur *cur, /* btree cursor */
|
struct xfs_btree_cur *cur, /* btree cursor */
|
||||||
int level, /* level in the btree */
|
int level, /* level in the btree */
|
||||||
@ -1846,7 +1844,7 @@ xfs_btree_lookup(
|
|||||||
int *stat) /* success/failure */
|
int *stat) /* success/failure */
|
||||||
{
|
{
|
||||||
struct xfs_btree_block *block; /* current btree block */
|
struct xfs_btree_block *block; /* current btree block */
|
||||||
__int64_t diff; /* difference for the current key */
|
int64_t diff; /* difference for the current key */
|
||||||
int error; /* error return value */
|
int error; /* error return value */
|
||||||
int keyno; /* current key number */
|
int keyno; /* current key number */
|
||||||
int level; /* level in the btree */
|
int level; /* level in the btree */
|
||||||
@ -4435,7 +4433,7 @@ xfs_btree_visit_blocks(
|
|||||||
* recovery completion writes the changes to disk.
|
* recovery completion writes the changes to disk.
|
||||||
*/
|
*/
|
||||||
struct xfs_btree_block_change_owner_info {
|
struct xfs_btree_block_change_owner_info {
|
||||||
__uint64_t new_owner;
|
uint64_t new_owner;
|
||||||
struct list_head *buffer_list;
|
struct list_head *buffer_list;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -4481,7 +4479,7 @@ xfs_btree_block_change_owner(
|
|||||||
int
|
int
|
||||||
xfs_btree_change_owner(
|
xfs_btree_change_owner(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
__uint64_t new_owner,
|
uint64_t new_owner,
|
||||||
struct list_head *buffer_list)
|
struct list_head *buffer_list)
|
||||||
{
|
{
|
||||||
struct xfs_btree_block_change_owner_info bbcoi;
|
struct xfs_btree_block_change_owner_info bbcoi;
|
||||||
@ -4585,7 +4583,7 @@ xfs_btree_simple_query_range(
|
|||||||
{
|
{
|
||||||
union xfs_btree_rec *recp;
|
union xfs_btree_rec *recp;
|
||||||
union xfs_btree_key rec_key;
|
union xfs_btree_key rec_key;
|
||||||
__int64_t diff;
|
int64_t diff;
|
||||||
int stat;
|
int stat;
|
||||||
bool firstrec = true;
|
bool firstrec = true;
|
||||||
int error;
|
int error;
|
||||||
@ -4682,8 +4680,8 @@ xfs_btree_overlapped_query_range(
|
|||||||
union xfs_btree_key *hkp;
|
union xfs_btree_key *hkp;
|
||||||
union xfs_btree_rec *recp;
|
union xfs_btree_rec *recp;
|
||||||
struct xfs_btree_block *block;
|
struct xfs_btree_block *block;
|
||||||
__int64_t ldiff;
|
int64_t ldiff;
|
||||||
__int64_t hdiff;
|
int64_t hdiff;
|
||||||
int level;
|
int level;
|
||||||
struct xfs_buf *bp;
|
struct xfs_buf *bp;
|
||||||
int i;
|
int i;
|
||||||
@ -4849,12 +4847,14 @@ xfs_btree_query_all(
|
|||||||
xfs_btree_query_range_fn fn,
|
xfs_btree_query_range_fn fn,
|
||||||
void *priv)
|
void *priv)
|
||||||
{
|
{
|
||||||
union xfs_btree_irec low_rec;
|
union xfs_btree_key low_key;
|
||||||
union xfs_btree_irec high_rec;
|
union xfs_btree_key high_key;
|
||||||
|
|
||||||
memset(&low_rec, 0, sizeof(low_rec));
|
memset(&cur->bc_rec, 0, sizeof(cur->bc_rec));
|
||||||
memset(&high_rec, 0xFF, sizeof(high_rec));
|
memset(&low_key, 0, sizeof(low_key));
|
||||||
return xfs_btree_query_range(cur, &low_rec, &high_rec, fn, priv);
|
memset(&high_key, 0xFF, sizeof(high_key));
|
||||||
|
|
||||||
|
return xfs_btree_simple_query_range(cur, &low_key, &high_key, fn, priv);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -76,7 +76,7 @@ union xfs_btree_rec {
|
|||||||
#define XFS_BTNUM_RMAP ((xfs_btnum_t)XFS_BTNUM_RMAPi)
|
#define XFS_BTNUM_RMAP ((xfs_btnum_t)XFS_BTNUM_RMAPi)
|
||||||
#define XFS_BTNUM_REFC ((xfs_btnum_t)XFS_BTNUM_REFCi)
|
#define XFS_BTNUM_REFC ((xfs_btnum_t)XFS_BTNUM_REFCi)
|
||||||
|
|
||||||
__uint32_t xfs_btree_magic(int crc, xfs_btnum_t btnum);
|
uint32_t xfs_btree_magic(int crc, xfs_btnum_t btnum);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For logging record fields.
|
* For logging record fields.
|
||||||
@ -150,20 +150,19 @@ struct xfs_btree_ops {
|
|||||||
union xfs_btree_rec *rec);
|
union xfs_btree_rec *rec);
|
||||||
|
|
||||||
/* difference between key value and cursor value */
|
/* difference between key value and cursor value */
|
||||||
__int64_t (*key_diff)(struct xfs_btree_cur *cur,
|
int64_t (*key_diff)(struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key);
|
union xfs_btree_key *key);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Difference between key2 and key1 -- positive if key1 > key2,
|
* Difference between key2 and key1 -- positive if key1 > key2,
|
||||||
* negative if key1 < key2, and zero if equal.
|
* negative if key1 < key2, and zero if equal.
|
||||||
*/
|
*/
|
||||||
__int64_t (*diff_two_keys)(struct xfs_btree_cur *cur,
|
int64_t (*diff_two_keys)(struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key1,
|
union xfs_btree_key *key1,
|
||||||
union xfs_btree_key *key2);
|
union xfs_btree_key *key2);
|
||||||
|
|
||||||
const struct xfs_buf_ops *buf_ops;
|
const struct xfs_buf_ops *buf_ops;
|
||||||
|
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
/* check that k1 is lower than k2 */
|
/* check that k1 is lower than k2 */
|
||||||
int (*keys_inorder)(struct xfs_btree_cur *cur,
|
int (*keys_inorder)(struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *k1,
|
union xfs_btree_key *k1,
|
||||||
@ -173,7 +172,6 @@ struct xfs_btree_ops {
|
|||||||
int (*recs_inorder)(struct xfs_btree_cur *cur,
|
int (*recs_inorder)(struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_rec *r1,
|
union xfs_btree_rec *r1,
|
||||||
union xfs_btree_rec *r2);
|
union xfs_btree_rec *r2);
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -213,11 +211,11 @@ typedef struct xfs_btree_cur
|
|||||||
union xfs_btree_irec bc_rec; /* current insert/search record value */
|
union xfs_btree_irec bc_rec; /* current insert/search record value */
|
||||||
struct xfs_buf *bc_bufs[XFS_BTREE_MAXLEVELS]; /* buf ptr per level */
|
struct xfs_buf *bc_bufs[XFS_BTREE_MAXLEVELS]; /* buf ptr per level */
|
||||||
int bc_ptrs[XFS_BTREE_MAXLEVELS]; /* key/record # */
|
int bc_ptrs[XFS_BTREE_MAXLEVELS]; /* key/record # */
|
||||||
__uint8_t bc_ra[XFS_BTREE_MAXLEVELS]; /* readahead bits */
|
uint8_t bc_ra[XFS_BTREE_MAXLEVELS]; /* readahead bits */
|
||||||
#define XFS_BTCUR_LEFTRA 1 /* left sibling has been read-ahead */
|
#define XFS_BTCUR_LEFTRA 1 /* left sibling has been read-ahead */
|
||||||
#define XFS_BTCUR_RIGHTRA 2 /* right sibling has been read-ahead */
|
#define XFS_BTCUR_RIGHTRA 2 /* right sibling has been read-ahead */
|
||||||
__uint8_t bc_nlevels; /* number of levels in the tree */
|
uint8_t bc_nlevels; /* number of levels in the tree */
|
||||||
__uint8_t bc_blocklog; /* log2(blocksize) of btree blocks */
|
uint8_t bc_blocklog; /* log2(blocksize) of btree blocks */
|
||||||
xfs_btnum_t bc_btnum; /* identifies which btree type */
|
xfs_btnum_t bc_btnum; /* identifies which btree type */
|
||||||
int bc_statoff; /* offset of btre stats array */
|
int bc_statoff; /* offset of btre stats array */
|
||||||
union {
|
union {
|
||||||
@ -330,7 +328,7 @@ xfs_btree_islastblock(
|
|||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
xfs_btree_offsets(
|
xfs_btree_offsets(
|
||||||
__int64_t fields, /* bitmask of fields */
|
int64_t fields, /* bitmask of fields */
|
||||||
const short *offsets,/* table of field offsets */
|
const short *offsets,/* table of field offsets */
|
||||||
int nbits, /* number of bits to inspect */
|
int nbits, /* number of bits to inspect */
|
||||||
int *first, /* output: first byte offset */
|
int *first, /* output: first byte offset */
|
||||||
@ -408,7 +406,7 @@ int xfs_btree_new_iroot(struct xfs_btree_cur *, int *, int *);
|
|||||||
int xfs_btree_insert(struct xfs_btree_cur *, int *);
|
int xfs_btree_insert(struct xfs_btree_cur *, int *);
|
||||||
int xfs_btree_delete(struct xfs_btree_cur *, int *);
|
int xfs_btree_delete(struct xfs_btree_cur *, int *);
|
||||||
int xfs_btree_get_rec(struct xfs_btree_cur *, union xfs_btree_rec **, int *);
|
int xfs_btree_get_rec(struct xfs_btree_cur *, union xfs_btree_rec **, int *);
|
||||||
int xfs_btree_change_owner(struct xfs_btree_cur *cur, __uint64_t new_owner,
|
int xfs_btree_change_owner(struct xfs_btree_cur *cur, uint64_t new_owner,
|
||||||
struct list_head *buffer_list);
|
struct list_head *buffer_list);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -434,7 +432,7 @@ static inline int xfs_btree_get_numrecs(struct xfs_btree_block *block)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline void xfs_btree_set_numrecs(struct xfs_btree_block *block,
|
static inline void xfs_btree_set_numrecs(struct xfs_btree_block *block,
|
||||||
__uint16_t numrecs)
|
uint16_t numrecs)
|
||||||
{
|
{
|
||||||
block->bb_numrecs = cpu_to_be16(numrecs);
|
block->bb_numrecs = cpu_to_be16(numrecs);
|
||||||
}
|
}
|
||||||
@ -506,4 +504,17 @@ int xfs_btree_visit_blocks(struct xfs_btree_cur *cur,
|
|||||||
|
|
||||||
int xfs_btree_count_blocks(struct xfs_btree_cur *cur, xfs_extlen_t *blocks);
|
int xfs_btree_count_blocks(struct xfs_btree_cur *cur, xfs_extlen_t *blocks);
|
||||||
|
|
||||||
|
union xfs_btree_rec *xfs_btree_rec_addr(struct xfs_btree_cur *cur, int n,
|
||||||
|
struct xfs_btree_block *block);
|
||||||
|
union xfs_btree_key *xfs_btree_key_addr(struct xfs_btree_cur *cur, int n,
|
||||||
|
struct xfs_btree_block *block);
|
||||||
|
union xfs_btree_key *xfs_btree_high_key_addr(struct xfs_btree_cur *cur, int n,
|
||||||
|
struct xfs_btree_block *block);
|
||||||
|
union xfs_btree_ptr *xfs_btree_ptr_addr(struct xfs_btree_cur *cur, int n,
|
||||||
|
struct xfs_btree_block *block);
|
||||||
|
int xfs_btree_lookup_get_block(struct xfs_btree_cur *cur, int level,
|
||||||
|
union xfs_btree_ptr *pp, struct xfs_btree_block **blkp);
|
||||||
|
struct xfs_btree_block *xfs_btree_get_block(struct xfs_btree_cur *cur,
|
||||||
|
int level, struct xfs_buf **bpp);
|
||||||
|
|
||||||
#endif /* __XFS_BTREE_H__ */
|
#endif /* __XFS_BTREE_H__ */
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#ifndef _XFS_CKSUM_H
|
#ifndef _XFS_CKSUM_H
|
||||||
#define _XFS_CKSUM_H 1
|
#define _XFS_CKSUM_H 1
|
||||||
|
|
||||||
#define XFS_CRC_SEED (~(__uint32_t)0)
|
#define XFS_CRC_SEED (~(uint32_t)0)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Calculate the intermediate checksum for a buffer that has the CRC field
|
* Calculate the intermediate checksum for a buffer that has the CRC field
|
||||||
@ -9,11 +9,11 @@
|
|||||||
* cksum_offset parameter. We do not modify the buffer during verification,
|
* cksum_offset parameter. We do not modify the buffer during verification,
|
||||||
* hence we have to split the CRC calculation across the cksum_offset.
|
* hence we have to split the CRC calculation across the cksum_offset.
|
||||||
*/
|
*/
|
||||||
static inline __uint32_t
|
static inline uint32_t
|
||||||
xfs_start_cksum_safe(char *buffer, size_t length, unsigned long cksum_offset)
|
xfs_start_cksum_safe(char *buffer, size_t length, unsigned long cksum_offset)
|
||||||
{
|
{
|
||||||
__uint32_t zero = 0;
|
uint32_t zero = 0;
|
||||||
__uint32_t crc;
|
uint32_t crc;
|
||||||
|
|
||||||
/* Calculate CRC up to the checksum. */
|
/* Calculate CRC up to the checksum. */
|
||||||
crc = crc32c(XFS_CRC_SEED, buffer, cksum_offset);
|
crc = crc32c(XFS_CRC_SEED, buffer, cksum_offset);
|
||||||
@ -30,7 +30,7 @@ xfs_start_cksum_safe(char *buffer, size_t length, unsigned long cksum_offset)
|
|||||||
* Fast CRC method where the buffer is modified. Callers must have exclusive
|
* Fast CRC method where the buffer is modified. Callers must have exclusive
|
||||||
* access to the buffer while the calculation takes place.
|
* access to the buffer while the calculation takes place.
|
||||||
*/
|
*/
|
||||||
static inline __uint32_t
|
static inline uint32_t
|
||||||
xfs_start_cksum_update(char *buffer, size_t length, unsigned long cksum_offset)
|
xfs_start_cksum_update(char *buffer, size_t length, unsigned long cksum_offset)
|
||||||
{
|
{
|
||||||
/* zero the CRC field */
|
/* zero the CRC field */
|
||||||
@ -48,7 +48,7 @@ xfs_start_cksum_update(char *buffer, size_t length, unsigned long cksum_offset)
|
|||||||
* so that it is consistent on disk.
|
* so that it is consistent on disk.
|
||||||
*/
|
*/
|
||||||
static inline __le32
|
static inline __le32
|
||||||
xfs_end_cksum(__uint32_t crc)
|
xfs_end_cksum(uint32_t crc)
|
||||||
{
|
{
|
||||||
return ~cpu_to_le32(crc);
|
return ~cpu_to_le32(crc);
|
||||||
}
|
}
|
||||||
@ -62,7 +62,7 @@ xfs_end_cksum(__uint32_t crc)
|
|||||||
static inline void
|
static inline void
|
||||||
xfs_update_cksum(char *buffer, size_t length, unsigned long cksum_offset)
|
xfs_update_cksum(char *buffer, size_t length, unsigned long cksum_offset)
|
||||||
{
|
{
|
||||||
__uint32_t crc = xfs_start_cksum_update(buffer, length, cksum_offset);
|
uint32_t crc = xfs_start_cksum_update(buffer, length, cksum_offset);
|
||||||
|
|
||||||
*(__le32 *)(buffer + cksum_offset) = xfs_end_cksum(crc);
|
*(__le32 *)(buffer + cksum_offset) = xfs_end_cksum(crc);
|
||||||
}
|
}
|
||||||
@ -73,7 +73,7 @@ xfs_update_cksum(char *buffer, size_t length, unsigned long cksum_offset)
|
|||||||
static inline int
|
static inline int
|
||||||
xfs_verify_cksum(char *buffer, size_t length, unsigned long cksum_offset)
|
xfs_verify_cksum(char *buffer, size_t length, unsigned long cksum_offset)
|
||||||
{
|
{
|
||||||
__uint32_t crc = xfs_start_cksum_safe(buffer, length, cksum_offset);
|
uint32_t crc = xfs_start_cksum_safe(buffer, length, cksum_offset);
|
||||||
|
|
||||||
return *(__le32 *)(buffer + cksum_offset) == xfs_end_cksum(crc);
|
return *(__le32 *)(buffer + cksum_offset) == xfs_end_cksum(crc);
|
||||||
}
|
}
|
||||||
|
@ -263,7 +263,7 @@ xfs_da3_node_read(
|
|||||||
|
|
||||||
err = xfs_da_read_buf(tp, dp, bno, mappedbno, bpp,
|
err = xfs_da_read_buf(tp, dp, bno, mappedbno, bpp,
|
||||||
which_fork, &xfs_da3_node_buf_ops);
|
which_fork, &xfs_da3_node_buf_ops);
|
||||||
if (!err && tp) {
|
if (!err && tp && *bpp) {
|
||||||
struct xfs_da_blkinfo *info = (*bpp)->b_addr;
|
struct xfs_da_blkinfo *info = (*bpp)->b_addr;
|
||||||
int type;
|
int type;
|
||||||
|
|
||||||
@ -1282,7 +1282,7 @@ xfs_da3_fixhashpath(
|
|||||||
return;
|
return;
|
||||||
break;
|
break;
|
||||||
case XFS_DIR2_LEAFN_MAGIC:
|
case XFS_DIR2_LEAFN_MAGIC:
|
||||||
lasthash = xfs_dir2_leafn_lasthash(dp, blk->bp, &count);
|
lasthash = xfs_dir2_leaf_lasthash(dp, blk->bp, &count);
|
||||||
if (count == 0)
|
if (count == 0)
|
||||||
return;
|
return;
|
||||||
break;
|
break;
|
||||||
@ -1502,8 +1502,8 @@ xfs_da3_node_lookup_int(
|
|||||||
if (blk->magic == XFS_DIR2_LEAFN_MAGIC ||
|
if (blk->magic == XFS_DIR2_LEAFN_MAGIC ||
|
||||||
blk->magic == XFS_DIR3_LEAFN_MAGIC) {
|
blk->magic == XFS_DIR3_LEAFN_MAGIC) {
|
||||||
blk->magic = XFS_DIR2_LEAFN_MAGIC;
|
blk->magic = XFS_DIR2_LEAFN_MAGIC;
|
||||||
blk->hashval = xfs_dir2_leafn_lasthash(args->dp,
|
blk->hashval = xfs_dir2_leaf_lasthash(args->dp,
|
||||||
blk->bp, NULL);
|
blk->bp, NULL);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1929,8 +1929,8 @@ xfs_da3_path_shift(
|
|||||||
blk->magic = XFS_DIR2_LEAFN_MAGIC;
|
blk->magic = XFS_DIR2_LEAFN_MAGIC;
|
||||||
ASSERT(level == path->active-1);
|
ASSERT(level == path->active-1);
|
||||||
blk->index = 0;
|
blk->index = 0;
|
||||||
blk->hashval = xfs_dir2_leafn_lasthash(args->dp,
|
blk->hashval = xfs_dir2_leaf_lasthash(args->dp,
|
||||||
blk->bp, NULL);
|
blk->bp, NULL);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
ASSERT(0);
|
ASSERT(0);
|
||||||
@ -1952,7 +1952,7 @@ xfs_da3_path_shift(
|
|||||||
* This is implemented with some source-level loop unrolling.
|
* This is implemented with some source-level loop unrolling.
|
||||||
*/
|
*/
|
||||||
xfs_dahash_t
|
xfs_dahash_t
|
||||||
xfs_da_hashname(const __uint8_t *name, int namelen)
|
xfs_da_hashname(const uint8_t *name, int namelen)
|
||||||
{
|
{
|
||||||
xfs_dahash_t hash;
|
xfs_dahash_t hash;
|
||||||
|
|
||||||
|
@ -60,10 +60,10 @@ enum xfs_dacmp {
|
|||||||
*/
|
*/
|
||||||
typedef struct xfs_da_args {
|
typedef struct xfs_da_args {
|
||||||
struct xfs_da_geometry *geo; /* da block geometry */
|
struct xfs_da_geometry *geo; /* da block geometry */
|
||||||
const __uint8_t *name; /* string (maybe not NULL terminated) */
|
const uint8_t *name; /* string (maybe not NULL terminated) */
|
||||||
int namelen; /* length of string (maybe no NULL) */
|
int namelen; /* length of string (maybe no NULL) */
|
||||||
__uint8_t filetype; /* filetype of inode for directories */
|
uint8_t filetype; /* filetype of inode for directories */
|
||||||
__uint8_t *value; /* set of bytes (maybe contain NULLs) */
|
uint8_t *value; /* set of bytes (maybe contain NULLs) */
|
||||||
int valuelen; /* length of value */
|
int valuelen; /* length of value */
|
||||||
int flags; /* argument flags (eg: ATTR_NOCREATE) */
|
int flags; /* argument flags (eg: ATTR_NOCREATE) */
|
||||||
xfs_dahash_t hashval; /* hash value of name */
|
xfs_dahash_t hashval; /* hash value of name */
|
||||||
@ -207,7 +207,7 @@ int xfs_da_reada_buf(struct xfs_inode *dp, xfs_dablk_t bno,
|
|||||||
int xfs_da_shrink_inode(xfs_da_args_t *args, xfs_dablk_t dead_blkno,
|
int xfs_da_shrink_inode(xfs_da_args_t *args, xfs_dablk_t dead_blkno,
|
||||||
struct xfs_buf *dead_buf);
|
struct xfs_buf *dead_buf);
|
||||||
|
|
||||||
uint xfs_da_hashname(const __uint8_t *name_string, int name_length);
|
uint xfs_da_hashname(const uint8_t *name_string, int name_length);
|
||||||
enum xfs_dacmp xfs_da_compname(struct xfs_da_args *args,
|
enum xfs_dacmp xfs_da_compname(struct xfs_da_args *args,
|
||||||
const unsigned char *name, int len);
|
const unsigned char *name, int len);
|
||||||
|
|
||||||
|
@ -49,7 +49,7 @@ xfs_dir3_sf_entsize(
|
|||||||
struct xfs_dir2_sf_hdr *hdr,
|
struct xfs_dir2_sf_hdr *hdr,
|
||||||
int len)
|
int len)
|
||||||
{
|
{
|
||||||
return xfs_dir2_sf_entsize(hdr, len) + sizeof(__uint8_t);
|
return xfs_dir2_sf_entsize(hdr, len) + sizeof(uint8_t);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct xfs_dir2_sf_entry *
|
static struct xfs_dir2_sf_entry *
|
||||||
@ -77,7 +77,7 @@ xfs_dir3_sf_nextentry(
|
|||||||
* not necessary. For non-filetype enable directories, the type is always
|
* not necessary. For non-filetype enable directories, the type is always
|
||||||
* unknown and we never store the value.
|
* unknown and we never store the value.
|
||||||
*/
|
*/
|
||||||
static __uint8_t
|
static uint8_t
|
||||||
xfs_dir2_sfe_get_ftype(
|
xfs_dir2_sfe_get_ftype(
|
||||||
struct xfs_dir2_sf_entry *sfep)
|
struct xfs_dir2_sf_entry *sfep)
|
||||||
{
|
{
|
||||||
@ -87,16 +87,16 @@ xfs_dir2_sfe_get_ftype(
|
|||||||
static void
|
static void
|
||||||
xfs_dir2_sfe_put_ftype(
|
xfs_dir2_sfe_put_ftype(
|
||||||
struct xfs_dir2_sf_entry *sfep,
|
struct xfs_dir2_sf_entry *sfep,
|
||||||
__uint8_t ftype)
|
uint8_t ftype)
|
||||||
{
|
{
|
||||||
ASSERT(ftype < XFS_DIR3_FT_MAX);
|
ASSERT(ftype < XFS_DIR3_FT_MAX);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __uint8_t
|
static uint8_t
|
||||||
xfs_dir3_sfe_get_ftype(
|
xfs_dir3_sfe_get_ftype(
|
||||||
struct xfs_dir2_sf_entry *sfep)
|
struct xfs_dir2_sf_entry *sfep)
|
||||||
{
|
{
|
||||||
__uint8_t ftype;
|
uint8_t ftype;
|
||||||
|
|
||||||
ftype = sfep->name[sfep->namelen];
|
ftype = sfep->name[sfep->namelen];
|
||||||
if (ftype >= XFS_DIR3_FT_MAX)
|
if (ftype >= XFS_DIR3_FT_MAX)
|
||||||
@ -107,7 +107,7 @@ xfs_dir3_sfe_get_ftype(
|
|||||||
static void
|
static void
|
||||||
xfs_dir3_sfe_put_ftype(
|
xfs_dir3_sfe_put_ftype(
|
||||||
struct xfs_dir2_sf_entry *sfep,
|
struct xfs_dir2_sf_entry *sfep,
|
||||||
__uint8_t ftype)
|
uint8_t ftype)
|
||||||
{
|
{
|
||||||
ASSERT(ftype < XFS_DIR3_FT_MAX);
|
ASSERT(ftype < XFS_DIR3_FT_MAX);
|
||||||
|
|
||||||
@ -124,7 +124,7 @@ xfs_dir3_sfe_put_ftype(
|
|||||||
static xfs_ino_t
|
static xfs_ino_t
|
||||||
xfs_dir2_sf_get_ino(
|
xfs_dir2_sf_get_ino(
|
||||||
struct xfs_dir2_sf_hdr *hdr,
|
struct xfs_dir2_sf_hdr *hdr,
|
||||||
__uint8_t *from)
|
uint8_t *from)
|
||||||
{
|
{
|
||||||
if (hdr->i8count)
|
if (hdr->i8count)
|
||||||
return get_unaligned_be64(from) & 0x00ffffffffffffffULL;
|
return get_unaligned_be64(from) & 0x00ffffffffffffffULL;
|
||||||
@ -135,7 +135,7 @@ xfs_dir2_sf_get_ino(
|
|||||||
static void
|
static void
|
||||||
xfs_dir2_sf_put_ino(
|
xfs_dir2_sf_put_ino(
|
||||||
struct xfs_dir2_sf_hdr *hdr,
|
struct xfs_dir2_sf_hdr *hdr,
|
||||||
__uint8_t *to,
|
uint8_t *to,
|
||||||
xfs_ino_t ino)
|
xfs_ino_t ino)
|
||||||
{
|
{
|
||||||
ASSERT((ino & 0xff00000000000000ULL) == 0);
|
ASSERT((ino & 0xff00000000000000ULL) == 0);
|
||||||
@ -225,7 +225,7 @@ xfs_dir3_sfe_put_ino(
|
|||||||
|
|
||||||
#define XFS_DIR3_DATA_ENTSIZE(n) \
|
#define XFS_DIR3_DATA_ENTSIZE(n) \
|
||||||
round_up((offsetof(struct xfs_dir2_data_entry, name[0]) + (n) + \
|
round_up((offsetof(struct xfs_dir2_data_entry, name[0]) + (n) + \
|
||||||
sizeof(xfs_dir2_data_off_t) + sizeof(__uint8_t)), \
|
sizeof(xfs_dir2_data_off_t) + sizeof(uint8_t)), \
|
||||||
XFS_DIR2_DATA_ALIGN)
|
XFS_DIR2_DATA_ALIGN)
|
||||||
|
|
||||||
static int
|
static int
|
||||||
@ -242,7 +242,7 @@ xfs_dir3_data_entsize(
|
|||||||
return XFS_DIR3_DATA_ENTSIZE(n);
|
return XFS_DIR3_DATA_ENTSIZE(n);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __uint8_t
|
static uint8_t
|
||||||
xfs_dir2_data_get_ftype(
|
xfs_dir2_data_get_ftype(
|
||||||
struct xfs_dir2_data_entry *dep)
|
struct xfs_dir2_data_entry *dep)
|
||||||
{
|
{
|
||||||
@ -252,16 +252,16 @@ xfs_dir2_data_get_ftype(
|
|||||||
static void
|
static void
|
||||||
xfs_dir2_data_put_ftype(
|
xfs_dir2_data_put_ftype(
|
||||||
struct xfs_dir2_data_entry *dep,
|
struct xfs_dir2_data_entry *dep,
|
||||||
__uint8_t ftype)
|
uint8_t ftype)
|
||||||
{
|
{
|
||||||
ASSERT(ftype < XFS_DIR3_FT_MAX);
|
ASSERT(ftype < XFS_DIR3_FT_MAX);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __uint8_t
|
static uint8_t
|
||||||
xfs_dir3_data_get_ftype(
|
xfs_dir3_data_get_ftype(
|
||||||
struct xfs_dir2_data_entry *dep)
|
struct xfs_dir2_data_entry *dep)
|
||||||
{
|
{
|
||||||
__uint8_t ftype = dep->name[dep->namelen];
|
uint8_t ftype = dep->name[dep->namelen];
|
||||||
|
|
||||||
if (ftype >= XFS_DIR3_FT_MAX)
|
if (ftype >= XFS_DIR3_FT_MAX)
|
||||||
return XFS_DIR3_FT_UNKNOWN;
|
return XFS_DIR3_FT_UNKNOWN;
|
||||||
@ -271,7 +271,7 @@ xfs_dir3_data_get_ftype(
|
|||||||
static void
|
static void
|
||||||
xfs_dir3_data_put_ftype(
|
xfs_dir3_data_put_ftype(
|
||||||
struct xfs_dir2_data_entry *dep,
|
struct xfs_dir2_data_entry *dep,
|
||||||
__uint8_t type)
|
uint8_t type)
|
||||||
{
|
{
|
||||||
ASSERT(type < XFS_DIR3_FT_MAX);
|
ASSERT(type < XFS_DIR3_FT_MAX);
|
||||||
ASSERT(dep->namelen != 0);
|
ASSERT(dep->namelen != 0);
|
||||||
|
@ -111,11 +111,11 @@ struct xfs_da3_intnode {
|
|||||||
* appropriate.
|
* appropriate.
|
||||||
*/
|
*/
|
||||||
struct xfs_da3_icnode_hdr {
|
struct xfs_da3_icnode_hdr {
|
||||||
__uint32_t forw;
|
uint32_t forw;
|
||||||
__uint32_t back;
|
uint32_t back;
|
||||||
__uint16_t magic;
|
uint16_t magic;
|
||||||
__uint16_t count;
|
uint16_t count;
|
||||||
__uint16_t level;
|
uint16_t level;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -187,14 +187,14 @@ struct xfs_da3_icnode_hdr {
|
|||||||
/*
|
/*
|
||||||
* Byte offset in data block and shortform entry.
|
* Byte offset in data block and shortform entry.
|
||||||
*/
|
*/
|
||||||
typedef __uint16_t xfs_dir2_data_off_t;
|
typedef uint16_t xfs_dir2_data_off_t;
|
||||||
#define NULLDATAOFF 0xffffU
|
#define NULLDATAOFF 0xffffU
|
||||||
typedef uint xfs_dir2_data_aoff_t; /* argument form */
|
typedef uint xfs_dir2_data_aoff_t; /* argument form */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Offset in data space of a data entry.
|
* Offset in data space of a data entry.
|
||||||
*/
|
*/
|
||||||
typedef __uint32_t xfs_dir2_dataptr_t;
|
typedef uint32_t xfs_dir2_dataptr_t;
|
||||||
#define XFS_DIR2_MAX_DATAPTR ((xfs_dir2_dataptr_t)0xffffffff)
|
#define XFS_DIR2_MAX_DATAPTR ((xfs_dir2_dataptr_t)0xffffffff)
|
||||||
#define XFS_DIR2_NULL_DATAPTR ((xfs_dir2_dataptr_t)0)
|
#define XFS_DIR2_NULL_DATAPTR ((xfs_dir2_dataptr_t)0)
|
||||||
|
|
||||||
@ -206,7 +206,7 @@ typedef xfs_off_t xfs_dir2_off_t;
|
|||||||
/*
|
/*
|
||||||
* Directory block number (logical dirblk in file)
|
* Directory block number (logical dirblk in file)
|
||||||
*/
|
*/
|
||||||
typedef __uint32_t xfs_dir2_db_t;
|
typedef uint32_t xfs_dir2_db_t;
|
||||||
|
|
||||||
#define XFS_INO32_SIZE 4
|
#define XFS_INO32_SIZE 4
|
||||||
#define XFS_INO64_SIZE 8
|
#define XFS_INO64_SIZE 8
|
||||||
@ -226,9 +226,9 @@ typedef __uint32_t xfs_dir2_db_t;
|
|||||||
* over them.
|
* over them.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_dir2_sf_hdr {
|
typedef struct xfs_dir2_sf_hdr {
|
||||||
__uint8_t count; /* count of entries */
|
uint8_t count; /* count of entries */
|
||||||
__uint8_t i8count; /* count of 8-byte inode #s */
|
uint8_t i8count; /* count of 8-byte inode #s */
|
||||||
__uint8_t parent[8]; /* parent dir inode number */
|
uint8_t parent[8]; /* parent dir inode number */
|
||||||
} __packed xfs_dir2_sf_hdr_t;
|
} __packed xfs_dir2_sf_hdr_t;
|
||||||
|
|
||||||
typedef struct xfs_dir2_sf_entry {
|
typedef struct xfs_dir2_sf_entry {
|
||||||
@ -447,11 +447,11 @@ struct xfs_dir3_leaf_hdr {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct xfs_dir3_icleaf_hdr {
|
struct xfs_dir3_icleaf_hdr {
|
||||||
__uint32_t forw;
|
uint32_t forw;
|
||||||
__uint32_t back;
|
uint32_t back;
|
||||||
__uint16_t magic;
|
uint16_t magic;
|
||||||
__uint16_t count;
|
uint16_t count;
|
||||||
__uint16_t stale;
|
uint16_t stale;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -538,10 +538,10 @@ struct xfs_dir3_free {
|
|||||||
* xfs_dir3_free_hdr_from_disk/xfs_dir3_free_hdr_to_disk.
|
* xfs_dir3_free_hdr_from_disk/xfs_dir3_free_hdr_to_disk.
|
||||||
*/
|
*/
|
||||||
struct xfs_dir3_icfree_hdr {
|
struct xfs_dir3_icfree_hdr {
|
||||||
__uint32_t magic;
|
uint32_t magic;
|
||||||
__uint32_t firstdb;
|
uint32_t firstdb;
|
||||||
__uint32_t nvalid;
|
uint32_t nvalid;
|
||||||
__uint32_t nused;
|
uint32_t nused;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -632,10 +632,10 @@ typedef struct xfs_attr_shortform {
|
|||||||
__u8 padding;
|
__u8 padding;
|
||||||
} hdr;
|
} hdr;
|
||||||
struct xfs_attr_sf_entry {
|
struct xfs_attr_sf_entry {
|
||||||
__uint8_t namelen; /* actual length of name (no NULL) */
|
uint8_t namelen; /* actual length of name (no NULL) */
|
||||||
__uint8_t valuelen; /* actual length of value (no NULL) */
|
uint8_t valuelen; /* actual length of value (no NULL) */
|
||||||
__uint8_t flags; /* flags bits (see xfs_attr_leaf.h) */
|
uint8_t flags; /* flags bits (see xfs_attr_leaf.h) */
|
||||||
__uint8_t nameval[1]; /* name & value bytes concatenated */
|
uint8_t nameval[1]; /* name & value bytes concatenated */
|
||||||
} list[1]; /* variable sized array */
|
} list[1]; /* variable sized array */
|
||||||
} xfs_attr_shortform_t;
|
} xfs_attr_shortform_t;
|
||||||
|
|
||||||
@ -725,22 +725,22 @@ struct xfs_attr3_leafblock {
|
|||||||
* incore, neutral version of the attribute leaf header
|
* incore, neutral version of the attribute leaf header
|
||||||
*/
|
*/
|
||||||
struct xfs_attr3_icleaf_hdr {
|
struct xfs_attr3_icleaf_hdr {
|
||||||
__uint32_t forw;
|
uint32_t forw;
|
||||||
__uint32_t back;
|
uint32_t back;
|
||||||
__uint16_t magic;
|
uint16_t magic;
|
||||||
__uint16_t count;
|
uint16_t count;
|
||||||
__uint16_t usedbytes;
|
uint16_t usedbytes;
|
||||||
/*
|
/*
|
||||||
* firstused is 32-bit here instead of 16-bit like the on-disk variant
|
* firstused is 32-bit here instead of 16-bit like the on-disk variant
|
||||||
* to support maximum fsb size of 64k without overflow issues throughout
|
* to support maximum fsb size of 64k without overflow issues throughout
|
||||||
* the attr code. Instead, the overflow condition is handled on
|
* the attr code. Instead, the overflow condition is handled on
|
||||||
* conversion to/from disk.
|
* conversion to/from disk.
|
||||||
*/
|
*/
|
||||||
__uint32_t firstused;
|
uint32_t firstused;
|
||||||
__u8 holes;
|
__u8 holes;
|
||||||
struct {
|
struct {
|
||||||
__uint16_t base;
|
uint16_t base;
|
||||||
__uint16_t size;
|
uint16_t size;
|
||||||
} freemap[XFS_ATTR_LEAF_MAPSIZE];
|
} freemap[XFS_ATTR_LEAF_MAPSIZE];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -218,8 +218,7 @@ xfs_dir_ino_validate(
|
|||||||
agblkno != 0 &&
|
agblkno != 0 &&
|
||||||
ioff < (1 << mp->m_sb.sb_inopblog) &&
|
ioff < (1 << mp->m_sb.sb_inopblog) &&
|
||||||
XFS_AGINO_TO_INO(mp, agno, agino) == ino;
|
XFS_AGINO_TO_INO(mp, agno, agino) == ino;
|
||||||
if (unlikely(XFS_TEST_ERROR(!ino_ok, mp, XFS_ERRTAG_DIR_INO_VALIDATE,
|
if (unlikely(XFS_TEST_ERROR(!ino_ok, mp, XFS_ERRTAG_DIR_INO_VALIDATE))) {
|
||||||
XFS_RANDOM_DIR_INO_VALIDATE))) {
|
|
||||||
xfs_warn(mp, "Invalid inode number 0x%Lx",
|
xfs_warn(mp, "Invalid inode number 0x%Lx",
|
||||||
(unsigned long long) ino);
|
(unsigned long long) ino);
|
||||||
XFS_ERROR_REPORT("xfs_dir_ino_validate", XFS_ERRLEVEL_LOW, mp);
|
XFS_ERROR_REPORT("xfs_dir_ino_validate", XFS_ERRLEVEL_LOW, mp);
|
||||||
|
@ -47,9 +47,9 @@ struct xfs_dir_ops {
|
|||||||
struct xfs_dir2_sf_entry *
|
struct xfs_dir2_sf_entry *
|
||||||
(*sf_nextentry)(struct xfs_dir2_sf_hdr *hdr,
|
(*sf_nextentry)(struct xfs_dir2_sf_hdr *hdr,
|
||||||
struct xfs_dir2_sf_entry *sfep);
|
struct xfs_dir2_sf_entry *sfep);
|
||||||
__uint8_t (*sf_get_ftype)(struct xfs_dir2_sf_entry *sfep);
|
uint8_t (*sf_get_ftype)(struct xfs_dir2_sf_entry *sfep);
|
||||||
void (*sf_put_ftype)(struct xfs_dir2_sf_entry *sfep,
|
void (*sf_put_ftype)(struct xfs_dir2_sf_entry *sfep,
|
||||||
__uint8_t ftype);
|
uint8_t ftype);
|
||||||
xfs_ino_t (*sf_get_ino)(struct xfs_dir2_sf_hdr *hdr,
|
xfs_ino_t (*sf_get_ino)(struct xfs_dir2_sf_hdr *hdr,
|
||||||
struct xfs_dir2_sf_entry *sfep);
|
struct xfs_dir2_sf_entry *sfep);
|
||||||
void (*sf_put_ino)(struct xfs_dir2_sf_hdr *hdr,
|
void (*sf_put_ino)(struct xfs_dir2_sf_hdr *hdr,
|
||||||
@ -60,9 +60,9 @@ struct xfs_dir_ops {
|
|||||||
xfs_ino_t ino);
|
xfs_ino_t ino);
|
||||||
|
|
||||||
int (*data_entsize)(int len);
|
int (*data_entsize)(int len);
|
||||||
__uint8_t (*data_get_ftype)(struct xfs_dir2_data_entry *dep);
|
uint8_t (*data_get_ftype)(struct xfs_dir2_data_entry *dep);
|
||||||
void (*data_put_ftype)(struct xfs_dir2_data_entry *dep,
|
void (*data_put_ftype)(struct xfs_dir2_data_entry *dep,
|
||||||
__uint8_t ftype);
|
uint8_t ftype);
|
||||||
__be16 * (*data_entry_tag_p)(struct xfs_dir2_data_entry *dep);
|
__be16 * (*data_entry_tag_p)(struct xfs_dir2_data_entry *dep);
|
||||||
struct xfs_dir2_data_free *
|
struct xfs_dir2_data_free *
|
||||||
(*data_bestfree_p)(struct xfs_dir2_data_hdr *hdr);
|
(*data_bestfree_p)(struct xfs_dir2_data_hdr *hdr);
|
||||||
|
@ -139,7 +139,7 @@ xfs_dir3_block_read(
|
|||||||
|
|
||||||
err = xfs_da_read_buf(tp, dp, mp->m_dir_geo->datablk, -1, bpp,
|
err = xfs_da_read_buf(tp, dp, mp->m_dir_geo->datablk, -1, bpp,
|
||||||
XFS_DATA_FORK, &xfs_dir3_block_buf_ops);
|
XFS_DATA_FORK, &xfs_dir3_block_buf_ops);
|
||||||
if (!err && tp)
|
if (!err && tp && *bpp)
|
||||||
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_DIR_BLOCK_BUF);
|
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_DIR_BLOCK_BUF);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
@ -145,7 +145,7 @@ xfs_dir3_leaf_check_int(
|
|||||||
static bool
|
static bool
|
||||||
xfs_dir3_leaf_verify(
|
xfs_dir3_leaf_verify(
|
||||||
struct xfs_buf *bp,
|
struct xfs_buf *bp,
|
||||||
__uint16_t magic)
|
uint16_t magic)
|
||||||
{
|
{
|
||||||
struct xfs_mount *mp = bp->b_target->bt_mount;
|
struct xfs_mount *mp = bp->b_target->bt_mount;
|
||||||
struct xfs_dir2_leaf *leaf = bp->b_addr;
|
struct xfs_dir2_leaf *leaf = bp->b_addr;
|
||||||
@ -154,7 +154,7 @@ xfs_dir3_leaf_verify(
|
|||||||
|
|
||||||
if (xfs_sb_version_hascrc(&mp->m_sb)) {
|
if (xfs_sb_version_hascrc(&mp->m_sb)) {
|
||||||
struct xfs_dir3_leaf_hdr *leaf3 = bp->b_addr;
|
struct xfs_dir3_leaf_hdr *leaf3 = bp->b_addr;
|
||||||
__uint16_t magic3;
|
uint16_t magic3;
|
||||||
|
|
||||||
magic3 = (magic == XFS_DIR2_LEAF1_MAGIC) ? XFS_DIR3_LEAF1_MAGIC
|
magic3 = (magic == XFS_DIR2_LEAF1_MAGIC) ? XFS_DIR3_LEAF1_MAGIC
|
||||||
: XFS_DIR3_LEAFN_MAGIC;
|
: XFS_DIR3_LEAFN_MAGIC;
|
||||||
@ -178,7 +178,7 @@ xfs_dir3_leaf_verify(
|
|||||||
static void
|
static void
|
||||||
__read_verify(
|
__read_verify(
|
||||||
struct xfs_buf *bp,
|
struct xfs_buf *bp,
|
||||||
__uint16_t magic)
|
uint16_t magic)
|
||||||
{
|
{
|
||||||
struct xfs_mount *mp = bp->b_target->bt_mount;
|
struct xfs_mount *mp = bp->b_target->bt_mount;
|
||||||
|
|
||||||
@ -195,7 +195,7 @@ __read_verify(
|
|||||||
static void
|
static void
|
||||||
__write_verify(
|
__write_verify(
|
||||||
struct xfs_buf *bp,
|
struct xfs_buf *bp,
|
||||||
__uint16_t magic)
|
uint16_t magic)
|
||||||
{
|
{
|
||||||
struct xfs_mount *mp = bp->b_target->bt_mount;
|
struct xfs_mount *mp = bp->b_target->bt_mount;
|
||||||
struct xfs_buf_log_item *bip = bp->b_fspriv;
|
struct xfs_buf_log_item *bip = bp->b_fspriv;
|
||||||
@ -256,7 +256,7 @@ const struct xfs_buf_ops xfs_dir3_leafn_buf_ops = {
|
|||||||
.verify_write = xfs_dir3_leafn_write_verify,
|
.verify_write = xfs_dir3_leafn_write_verify,
|
||||||
};
|
};
|
||||||
|
|
||||||
static int
|
int
|
||||||
xfs_dir3_leaf_read(
|
xfs_dir3_leaf_read(
|
||||||
struct xfs_trans *tp,
|
struct xfs_trans *tp,
|
||||||
struct xfs_inode *dp,
|
struct xfs_inode *dp,
|
||||||
@ -268,7 +268,7 @@ xfs_dir3_leaf_read(
|
|||||||
|
|
||||||
err = xfs_da_read_buf(tp, dp, fbno, mappedbno, bpp,
|
err = xfs_da_read_buf(tp, dp, fbno, mappedbno, bpp,
|
||||||
XFS_DATA_FORK, &xfs_dir3_leaf1_buf_ops);
|
XFS_DATA_FORK, &xfs_dir3_leaf1_buf_ops);
|
||||||
if (!err && tp)
|
if (!err && tp && *bpp)
|
||||||
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_DIR_LEAF1_BUF);
|
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_DIR_LEAF1_BUF);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
@ -285,7 +285,7 @@ xfs_dir3_leafn_read(
|
|||||||
|
|
||||||
err = xfs_da_read_buf(tp, dp, fbno, mappedbno, bpp,
|
err = xfs_da_read_buf(tp, dp, fbno, mappedbno, bpp,
|
||||||
XFS_DATA_FORK, &xfs_dir3_leafn_buf_ops);
|
XFS_DATA_FORK, &xfs_dir3_leafn_buf_ops);
|
||||||
if (!err && tp)
|
if (!err && tp && *bpp)
|
||||||
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_DIR_LEAFN_BUF);
|
xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_DIR_LEAFN_BUF);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
@ -299,7 +299,7 @@ xfs_dir3_leaf_init(
|
|||||||
struct xfs_trans *tp,
|
struct xfs_trans *tp,
|
||||||
struct xfs_buf *bp,
|
struct xfs_buf *bp,
|
||||||
xfs_ino_t owner,
|
xfs_ino_t owner,
|
||||||
__uint16_t type)
|
uint16_t type)
|
||||||
{
|
{
|
||||||
struct xfs_dir2_leaf *leaf = bp->b_addr;
|
struct xfs_dir2_leaf *leaf = bp->b_addr;
|
||||||
|
|
||||||
@ -343,7 +343,7 @@ xfs_dir3_leaf_get_buf(
|
|||||||
xfs_da_args_t *args,
|
xfs_da_args_t *args,
|
||||||
xfs_dir2_db_t bno,
|
xfs_dir2_db_t bno,
|
||||||
struct xfs_buf **bpp,
|
struct xfs_buf **bpp,
|
||||||
__uint16_t magic)
|
uint16_t magic)
|
||||||
{
|
{
|
||||||
struct xfs_inode *dp = args->dp;
|
struct xfs_inode *dp = args->dp;
|
||||||
struct xfs_trans *tp = args->trans;
|
struct xfs_trans *tp = args->trans;
|
||||||
|
@ -528,7 +528,7 @@ xfs_dir2_free_hdr_check(
|
|||||||
* Stale entries are ok.
|
* Stale entries are ok.
|
||||||
*/
|
*/
|
||||||
xfs_dahash_t /* hash value */
|
xfs_dahash_t /* hash value */
|
||||||
xfs_dir2_leafn_lasthash(
|
xfs_dir2_leaf_lasthash(
|
||||||
struct xfs_inode *dp,
|
struct xfs_inode *dp,
|
||||||
struct xfs_buf *bp, /* leaf buffer */
|
struct xfs_buf *bp, /* leaf buffer */
|
||||||
int *count) /* count of entries in leaf */
|
int *count) /* count of entries in leaf */
|
||||||
@ -540,7 +540,9 @@ xfs_dir2_leafn_lasthash(
|
|||||||
dp->d_ops->leaf_hdr_from_disk(&leafhdr, leaf);
|
dp->d_ops->leaf_hdr_from_disk(&leafhdr, leaf);
|
||||||
|
|
||||||
ASSERT(leafhdr.magic == XFS_DIR2_LEAFN_MAGIC ||
|
ASSERT(leafhdr.magic == XFS_DIR2_LEAFN_MAGIC ||
|
||||||
leafhdr.magic == XFS_DIR3_LEAFN_MAGIC);
|
leafhdr.magic == XFS_DIR3_LEAFN_MAGIC ||
|
||||||
|
leafhdr.magic == XFS_DIR2_LEAF1_MAGIC ||
|
||||||
|
leafhdr.magic == XFS_DIR3_LEAF1_MAGIC);
|
||||||
|
|
||||||
if (count)
|
if (count)
|
||||||
*count = leafhdr.count;
|
*count = leafhdr.count;
|
||||||
@ -1405,8 +1407,8 @@ xfs_dir2_leafn_split(
|
|||||||
/*
|
/*
|
||||||
* Update last hashval in each block since we added the name.
|
* Update last hashval in each block since we added the name.
|
||||||
*/
|
*/
|
||||||
oldblk->hashval = xfs_dir2_leafn_lasthash(dp, oldblk->bp, NULL);
|
oldblk->hashval = xfs_dir2_leaf_lasthash(dp, oldblk->bp, NULL);
|
||||||
newblk->hashval = xfs_dir2_leafn_lasthash(dp, newblk->bp, NULL);
|
newblk->hashval = xfs_dir2_leaf_lasthash(dp, newblk->bp, NULL);
|
||||||
xfs_dir3_leaf_check(dp, oldblk->bp);
|
xfs_dir3_leaf_check(dp, oldblk->bp);
|
||||||
xfs_dir3_leaf_check(dp, newblk->bp);
|
xfs_dir3_leaf_check(dp, newblk->bp);
|
||||||
return error;
|
return error;
|
||||||
|
@ -58,6 +58,8 @@ extern int xfs_dir3_data_init(struct xfs_da_args *args, xfs_dir2_db_t blkno,
|
|||||||
struct xfs_buf **bpp);
|
struct xfs_buf **bpp);
|
||||||
|
|
||||||
/* xfs_dir2_leaf.c */
|
/* xfs_dir2_leaf.c */
|
||||||
|
extern int xfs_dir3_leaf_read(struct xfs_trans *tp, struct xfs_inode *dp,
|
||||||
|
xfs_dablk_t fbno, xfs_daddr_t mappedbno, struct xfs_buf **bpp);
|
||||||
extern int xfs_dir3_leafn_read(struct xfs_trans *tp, struct xfs_inode *dp,
|
extern int xfs_dir3_leafn_read(struct xfs_trans *tp, struct xfs_inode *dp,
|
||||||
xfs_dablk_t fbno, xfs_daddr_t mappedbno, struct xfs_buf **bpp);
|
xfs_dablk_t fbno, xfs_daddr_t mappedbno, struct xfs_buf **bpp);
|
||||||
extern int xfs_dir2_block_to_leaf(struct xfs_da_args *args,
|
extern int xfs_dir2_block_to_leaf(struct xfs_da_args *args,
|
||||||
@ -69,7 +71,7 @@ extern void xfs_dir3_leaf_compact_x1(struct xfs_dir3_icleaf_hdr *leafhdr,
|
|||||||
struct xfs_dir2_leaf_entry *ents, int *indexp,
|
struct xfs_dir2_leaf_entry *ents, int *indexp,
|
||||||
int *lowstalep, int *highstalep, int *lowlogp, int *highlogp);
|
int *lowstalep, int *highstalep, int *lowlogp, int *highlogp);
|
||||||
extern int xfs_dir3_leaf_get_buf(struct xfs_da_args *args, xfs_dir2_db_t bno,
|
extern int xfs_dir3_leaf_get_buf(struct xfs_da_args *args, xfs_dir2_db_t bno,
|
||||||
struct xfs_buf **bpp, __uint16_t magic);
|
struct xfs_buf **bpp, uint16_t magic);
|
||||||
extern void xfs_dir3_leaf_log_ents(struct xfs_da_args *args,
|
extern void xfs_dir3_leaf_log_ents(struct xfs_da_args *args,
|
||||||
struct xfs_buf *bp, int first, int last);
|
struct xfs_buf *bp, int first, int last);
|
||||||
extern void xfs_dir3_leaf_log_header(struct xfs_da_args *args,
|
extern void xfs_dir3_leaf_log_header(struct xfs_da_args *args,
|
||||||
@ -93,7 +95,7 @@ extern bool xfs_dir3_leaf_check_int(struct xfs_mount *mp, struct xfs_inode *dp,
|
|||||||
/* xfs_dir2_node.c */
|
/* xfs_dir2_node.c */
|
||||||
extern int xfs_dir2_leaf_to_node(struct xfs_da_args *args,
|
extern int xfs_dir2_leaf_to_node(struct xfs_da_args *args,
|
||||||
struct xfs_buf *lbp);
|
struct xfs_buf *lbp);
|
||||||
extern xfs_dahash_t xfs_dir2_leafn_lasthash(struct xfs_inode *dp,
|
extern xfs_dahash_t xfs_dir2_leaf_lasthash(struct xfs_inode *dp,
|
||||||
struct xfs_buf *bp, int *count);
|
struct xfs_buf *bp, int *count);
|
||||||
extern int xfs_dir2_leafn_lookup_int(struct xfs_buf *bp,
|
extern int xfs_dir2_leafn_lookup_int(struct xfs_buf *bp,
|
||||||
struct xfs_da_args *args, int *indexp,
|
struct xfs_da_args *args, int *indexp,
|
||||||
@ -128,7 +130,7 @@ extern int xfs_dir2_sf_replace(struct xfs_da_args *args);
|
|||||||
extern int xfs_dir2_sf_verify(struct xfs_inode *ip);
|
extern int xfs_dir2_sf_verify(struct xfs_inode *ip);
|
||||||
|
|
||||||
/* xfs_dir2_readdir.c */
|
/* xfs_dir2_readdir.c */
|
||||||
extern int xfs_readdir(struct xfs_inode *dp, struct dir_context *ctx,
|
extern int xfs_readdir(struct xfs_trans *tp, struct xfs_inode *dp,
|
||||||
size_t bufsize);
|
struct dir_context *ctx, size_t bufsize);
|
||||||
|
|
||||||
#endif /* __XFS_DIR2_PRIV_H__ */
|
#endif /* __XFS_DIR2_PRIV_H__ */
|
||||||
|
@ -647,7 +647,7 @@ xfs_dir2_sf_verify(
|
|||||||
int offset;
|
int offset;
|
||||||
int size;
|
int size;
|
||||||
int error;
|
int error;
|
||||||
__uint8_t filetype;
|
uint8_t filetype;
|
||||||
|
|
||||||
ASSERT(ip->i_d.di_format == XFS_DINODE_FMT_LOCAL);
|
ASSERT(ip->i_d.di_format == XFS_DINODE_FMT_LOCAL);
|
||||||
/*
|
/*
|
||||||
|
@ -103,8 +103,8 @@ struct xfs_ifork;
|
|||||||
* Must be padded to 64 bit alignment.
|
* Must be padded to 64 bit alignment.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_sb {
|
typedef struct xfs_sb {
|
||||||
__uint32_t sb_magicnum; /* magic number == XFS_SB_MAGIC */
|
uint32_t sb_magicnum; /* magic number == XFS_SB_MAGIC */
|
||||||
__uint32_t sb_blocksize; /* logical block size, bytes */
|
uint32_t sb_blocksize; /* logical block size, bytes */
|
||||||
xfs_rfsblock_t sb_dblocks; /* number of data blocks */
|
xfs_rfsblock_t sb_dblocks; /* number of data blocks */
|
||||||
xfs_rfsblock_t sb_rblocks; /* number of realtime blocks */
|
xfs_rfsblock_t sb_rblocks; /* number of realtime blocks */
|
||||||
xfs_rtblock_t sb_rextents; /* number of realtime extents */
|
xfs_rtblock_t sb_rextents; /* number of realtime extents */
|
||||||
@ -118,45 +118,45 @@ typedef struct xfs_sb {
|
|||||||
xfs_agnumber_t sb_agcount; /* number of allocation groups */
|
xfs_agnumber_t sb_agcount; /* number of allocation groups */
|
||||||
xfs_extlen_t sb_rbmblocks; /* number of rt bitmap blocks */
|
xfs_extlen_t sb_rbmblocks; /* number of rt bitmap blocks */
|
||||||
xfs_extlen_t sb_logblocks; /* number of log blocks */
|
xfs_extlen_t sb_logblocks; /* number of log blocks */
|
||||||
__uint16_t sb_versionnum; /* header version == XFS_SB_VERSION */
|
uint16_t sb_versionnum; /* header version == XFS_SB_VERSION */
|
||||||
__uint16_t sb_sectsize; /* volume sector size, bytes */
|
uint16_t sb_sectsize; /* volume sector size, bytes */
|
||||||
__uint16_t sb_inodesize; /* inode size, bytes */
|
uint16_t sb_inodesize; /* inode size, bytes */
|
||||||
__uint16_t sb_inopblock; /* inodes per block */
|
uint16_t sb_inopblock; /* inodes per block */
|
||||||
char sb_fname[12]; /* file system name */
|
char sb_fname[12]; /* file system name */
|
||||||
__uint8_t sb_blocklog; /* log2 of sb_blocksize */
|
uint8_t sb_blocklog; /* log2 of sb_blocksize */
|
||||||
__uint8_t sb_sectlog; /* log2 of sb_sectsize */
|
uint8_t sb_sectlog; /* log2 of sb_sectsize */
|
||||||
__uint8_t sb_inodelog; /* log2 of sb_inodesize */
|
uint8_t sb_inodelog; /* log2 of sb_inodesize */
|
||||||
__uint8_t sb_inopblog; /* log2 of sb_inopblock */
|
uint8_t sb_inopblog; /* log2 of sb_inopblock */
|
||||||
__uint8_t sb_agblklog; /* log2 of sb_agblocks (rounded up) */
|
uint8_t sb_agblklog; /* log2 of sb_agblocks (rounded up) */
|
||||||
__uint8_t sb_rextslog; /* log2 of sb_rextents */
|
uint8_t sb_rextslog; /* log2 of sb_rextents */
|
||||||
__uint8_t sb_inprogress; /* mkfs is in progress, don't mount */
|
uint8_t sb_inprogress; /* mkfs is in progress, don't mount */
|
||||||
__uint8_t sb_imax_pct; /* max % of fs for inode space */
|
uint8_t sb_imax_pct; /* max % of fs for inode space */
|
||||||
/* statistics */
|
/* statistics */
|
||||||
/*
|
/*
|
||||||
* These fields must remain contiguous. If you really
|
* These fields must remain contiguous. If you really
|
||||||
* want to change their layout, make sure you fix the
|
* want to change their layout, make sure you fix the
|
||||||
* code in xfs_trans_apply_sb_deltas().
|
* code in xfs_trans_apply_sb_deltas().
|
||||||
*/
|
*/
|
||||||
__uint64_t sb_icount; /* allocated inodes */
|
uint64_t sb_icount; /* allocated inodes */
|
||||||
__uint64_t sb_ifree; /* free inodes */
|
uint64_t sb_ifree; /* free inodes */
|
||||||
__uint64_t sb_fdblocks; /* free data blocks */
|
uint64_t sb_fdblocks; /* free data blocks */
|
||||||
__uint64_t sb_frextents; /* free realtime extents */
|
uint64_t sb_frextents; /* free realtime extents */
|
||||||
/*
|
/*
|
||||||
* End contiguous fields.
|
* End contiguous fields.
|
||||||
*/
|
*/
|
||||||
xfs_ino_t sb_uquotino; /* user quota inode */
|
xfs_ino_t sb_uquotino; /* user quota inode */
|
||||||
xfs_ino_t sb_gquotino; /* group quota inode */
|
xfs_ino_t sb_gquotino; /* group quota inode */
|
||||||
__uint16_t sb_qflags; /* quota flags */
|
uint16_t sb_qflags; /* quota flags */
|
||||||
__uint8_t sb_flags; /* misc. flags */
|
uint8_t sb_flags; /* misc. flags */
|
||||||
__uint8_t sb_shared_vn; /* shared version number */
|
uint8_t sb_shared_vn; /* shared version number */
|
||||||
xfs_extlen_t sb_inoalignmt; /* inode chunk alignment, fsblocks */
|
xfs_extlen_t sb_inoalignmt; /* inode chunk alignment, fsblocks */
|
||||||
__uint32_t sb_unit; /* stripe or raid unit */
|
uint32_t sb_unit; /* stripe or raid unit */
|
||||||
__uint32_t sb_width; /* stripe or raid width */
|
uint32_t sb_width; /* stripe or raid width */
|
||||||
__uint8_t sb_dirblklog; /* log2 of dir block size (fsbs) */
|
uint8_t sb_dirblklog; /* log2 of dir block size (fsbs) */
|
||||||
__uint8_t sb_logsectlog; /* log2 of the log sector size */
|
uint8_t sb_logsectlog; /* log2 of the log sector size */
|
||||||
__uint16_t sb_logsectsize; /* sector size for the log, bytes */
|
uint16_t sb_logsectsize; /* sector size for the log, bytes */
|
||||||
__uint32_t sb_logsunit; /* stripe unit size for the log */
|
uint32_t sb_logsunit; /* stripe unit size for the log */
|
||||||
__uint32_t sb_features2; /* additional feature bits */
|
uint32_t sb_features2; /* additional feature bits */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* bad features2 field as a result of failing to pad the sb structure to
|
* bad features2 field as a result of failing to pad the sb structure to
|
||||||
@ -167,17 +167,17 @@ typedef struct xfs_sb {
|
|||||||
* the value in sb_features2 when formatting the incore superblock to
|
* the value in sb_features2 when formatting the incore superblock to
|
||||||
* the disk buffer.
|
* the disk buffer.
|
||||||
*/
|
*/
|
||||||
__uint32_t sb_bad_features2;
|
uint32_t sb_bad_features2;
|
||||||
|
|
||||||
/* version 5 superblock fields start here */
|
/* version 5 superblock fields start here */
|
||||||
|
|
||||||
/* feature masks */
|
/* feature masks */
|
||||||
__uint32_t sb_features_compat;
|
uint32_t sb_features_compat;
|
||||||
__uint32_t sb_features_ro_compat;
|
uint32_t sb_features_ro_compat;
|
||||||
__uint32_t sb_features_incompat;
|
uint32_t sb_features_incompat;
|
||||||
__uint32_t sb_features_log_incompat;
|
uint32_t sb_features_log_incompat;
|
||||||
|
|
||||||
__uint32_t sb_crc; /* superblock crc */
|
uint32_t sb_crc; /* superblock crc */
|
||||||
xfs_extlen_t sb_spino_align; /* sparse inode chunk alignment */
|
xfs_extlen_t sb_spino_align; /* sparse inode chunk alignment */
|
||||||
|
|
||||||
xfs_ino_t sb_pquotino; /* project quota inode */
|
xfs_ino_t sb_pquotino; /* project quota inode */
|
||||||
@ -449,7 +449,7 @@ static inline void xfs_sb_version_addprojid32bit(struct xfs_sb *sbp)
|
|||||||
static inline bool
|
static inline bool
|
||||||
xfs_sb_has_compat_feature(
|
xfs_sb_has_compat_feature(
|
||||||
struct xfs_sb *sbp,
|
struct xfs_sb *sbp,
|
||||||
__uint32_t feature)
|
uint32_t feature)
|
||||||
{
|
{
|
||||||
return (sbp->sb_features_compat & feature) != 0;
|
return (sbp->sb_features_compat & feature) != 0;
|
||||||
}
|
}
|
||||||
@ -465,7 +465,7 @@ xfs_sb_has_compat_feature(
|
|||||||
static inline bool
|
static inline bool
|
||||||
xfs_sb_has_ro_compat_feature(
|
xfs_sb_has_ro_compat_feature(
|
||||||
struct xfs_sb *sbp,
|
struct xfs_sb *sbp,
|
||||||
__uint32_t feature)
|
uint32_t feature)
|
||||||
{
|
{
|
||||||
return (sbp->sb_features_ro_compat & feature) != 0;
|
return (sbp->sb_features_ro_compat & feature) != 0;
|
||||||
}
|
}
|
||||||
@ -482,7 +482,7 @@ xfs_sb_has_ro_compat_feature(
|
|||||||
static inline bool
|
static inline bool
|
||||||
xfs_sb_has_incompat_feature(
|
xfs_sb_has_incompat_feature(
|
||||||
struct xfs_sb *sbp,
|
struct xfs_sb *sbp,
|
||||||
__uint32_t feature)
|
uint32_t feature)
|
||||||
{
|
{
|
||||||
return (sbp->sb_features_incompat & feature) != 0;
|
return (sbp->sb_features_incompat & feature) != 0;
|
||||||
}
|
}
|
||||||
@ -492,7 +492,7 @@ xfs_sb_has_incompat_feature(
|
|||||||
static inline bool
|
static inline bool
|
||||||
xfs_sb_has_incompat_log_feature(
|
xfs_sb_has_incompat_log_feature(
|
||||||
struct xfs_sb *sbp,
|
struct xfs_sb *sbp,
|
||||||
__uint32_t feature)
|
uint32_t feature)
|
||||||
{
|
{
|
||||||
return (sbp->sb_features_log_incompat & feature) != 0;
|
return (sbp->sb_features_log_incompat & feature) != 0;
|
||||||
}
|
}
|
||||||
@ -594,8 +594,8 @@ xfs_is_quota_inode(struct xfs_sb *sbp, xfs_ino_t ino)
|
|||||||
*/
|
*/
|
||||||
#define XFS_FSB_TO_B(mp,fsbno) ((xfs_fsize_t)(fsbno) << (mp)->m_sb.sb_blocklog)
|
#define XFS_FSB_TO_B(mp,fsbno) ((xfs_fsize_t)(fsbno) << (mp)->m_sb.sb_blocklog)
|
||||||
#define XFS_B_TO_FSB(mp,b) \
|
#define XFS_B_TO_FSB(mp,b) \
|
||||||
((((__uint64_t)(b)) + (mp)->m_blockmask) >> (mp)->m_sb.sb_blocklog)
|
((((uint64_t)(b)) + (mp)->m_blockmask) >> (mp)->m_sb.sb_blocklog)
|
||||||
#define XFS_B_TO_FSBT(mp,b) (((__uint64_t)(b)) >> (mp)->m_sb.sb_blocklog)
|
#define XFS_B_TO_FSBT(mp,b) (((uint64_t)(b)) >> (mp)->m_sb.sb_blocklog)
|
||||||
#define XFS_B_FSB_OFFSET(mp,b) ((b) & (mp)->m_blockmask)
|
#define XFS_B_FSB_OFFSET(mp,b) ((b) & (mp)->m_blockmask)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1072,7 +1072,7 @@ static inline void xfs_dinode_put_rdev(struct xfs_dinode *dip, xfs_dev_t rdev)
|
|||||||
* next agno_log bits - ag number
|
* next agno_log bits - ag number
|
||||||
* high agno_log-agblklog-inopblog bits - 0
|
* high agno_log-agblklog-inopblog bits - 0
|
||||||
*/
|
*/
|
||||||
#define XFS_INO_MASK(k) (__uint32_t)((1ULL << (k)) - 1)
|
#define XFS_INO_MASK(k) (uint32_t)((1ULL << (k)) - 1)
|
||||||
#define XFS_INO_OFFSET_BITS(mp) (mp)->m_sb.sb_inopblog
|
#define XFS_INO_OFFSET_BITS(mp) (mp)->m_sb.sb_inopblog
|
||||||
#define XFS_INO_AGBNO_BITS(mp) (mp)->m_sb.sb_agblklog
|
#define XFS_INO_AGBNO_BITS(mp) (mp)->m_sb.sb_agblklog
|
||||||
#define XFS_INO_AGINO_BITS(mp) (mp)->m_agino_log
|
#define XFS_INO_AGINO_BITS(mp) (mp)->m_agino_log
|
||||||
@ -1211,6 +1211,7 @@ struct xfs_dsymlink_hdr {
|
|||||||
|
|
||||||
#define XFS_SYMLINK_CRC_OFF offsetof(struct xfs_dsymlink_hdr, sl_crc)
|
#define XFS_SYMLINK_CRC_OFF offsetof(struct xfs_dsymlink_hdr, sl_crc)
|
||||||
|
|
||||||
|
#define XFS_SYMLINK_MAXLEN 1024
|
||||||
/*
|
/*
|
||||||
* The maximum pathlen is 1024 bytes. Since the minimum file system
|
* The maximum pathlen is 1024 bytes. Since the minimum file system
|
||||||
* blocksize is 512 bytes, we can get a max of 3 extents back from
|
* blocksize is 512 bytes, we can get a max of 3 extents back from
|
||||||
@ -1269,16 +1270,16 @@ typedef __be32 xfs_alloc_ptr_t;
|
|||||||
#define XFS_FIBT_MAGIC 0x46494254 /* 'FIBT' */
|
#define XFS_FIBT_MAGIC 0x46494254 /* 'FIBT' */
|
||||||
#define XFS_FIBT_CRC_MAGIC 0x46494233 /* 'FIB3' */
|
#define XFS_FIBT_CRC_MAGIC 0x46494233 /* 'FIB3' */
|
||||||
|
|
||||||
typedef __uint64_t xfs_inofree_t;
|
typedef uint64_t xfs_inofree_t;
|
||||||
#define XFS_INODES_PER_CHUNK (NBBY * sizeof(xfs_inofree_t))
|
#define XFS_INODES_PER_CHUNK (NBBY * sizeof(xfs_inofree_t))
|
||||||
#define XFS_INODES_PER_CHUNK_LOG (XFS_NBBYLOG + 3)
|
#define XFS_INODES_PER_CHUNK_LOG (XFS_NBBYLOG + 3)
|
||||||
#define XFS_INOBT_ALL_FREE ((xfs_inofree_t)-1)
|
#define XFS_INOBT_ALL_FREE ((xfs_inofree_t)-1)
|
||||||
#define XFS_INOBT_MASK(i) ((xfs_inofree_t)1 << (i))
|
#define XFS_INOBT_MASK(i) ((xfs_inofree_t)1 << (i))
|
||||||
|
|
||||||
#define XFS_INOBT_HOLEMASK_FULL 0 /* holemask for full chunk */
|
#define XFS_INOBT_HOLEMASK_FULL 0 /* holemask for full chunk */
|
||||||
#define XFS_INOBT_HOLEMASK_BITS (NBBY * sizeof(__uint16_t))
|
#define XFS_INOBT_HOLEMASK_BITS (NBBY * sizeof(uint16_t))
|
||||||
#define XFS_INODES_PER_HOLEMASK_BIT \
|
#define XFS_INODES_PER_HOLEMASK_BIT \
|
||||||
(XFS_INODES_PER_CHUNK / (NBBY * sizeof(__uint16_t)))
|
(XFS_INODES_PER_CHUNK / (NBBY * sizeof(uint16_t)))
|
||||||
|
|
||||||
static inline xfs_inofree_t xfs_inobt_maskn(int i, int n)
|
static inline xfs_inofree_t xfs_inobt_maskn(int i, int n)
|
||||||
{
|
{
|
||||||
@ -1312,9 +1313,9 @@ typedef struct xfs_inobt_rec {
|
|||||||
|
|
||||||
typedef struct xfs_inobt_rec_incore {
|
typedef struct xfs_inobt_rec_incore {
|
||||||
xfs_agino_t ir_startino; /* starting inode number */
|
xfs_agino_t ir_startino; /* starting inode number */
|
||||||
__uint16_t ir_holemask; /* hole mask for sparse chunks */
|
uint16_t ir_holemask; /* hole mask for sparse chunks */
|
||||||
__uint8_t ir_count; /* total inode count */
|
uint8_t ir_count; /* total inode count */
|
||||||
__uint8_t ir_freecount; /* count of free inodes (set bits) */
|
uint8_t ir_freecount; /* count of free inodes (set bits) */
|
||||||
xfs_inofree_t ir_free; /* free inode mask */
|
xfs_inofree_t ir_free; /* free inode mask */
|
||||||
} xfs_inobt_rec_incore_t;
|
} xfs_inobt_rec_incore_t;
|
||||||
|
|
||||||
@ -1397,15 +1398,15 @@ struct xfs_rmap_rec {
|
|||||||
* rm_offset:54-60 aren't used and should be zero
|
* rm_offset:54-60 aren't used and should be zero
|
||||||
* rm_offset:0-53 is the block offset within the inode
|
* rm_offset:0-53 is the block offset within the inode
|
||||||
*/
|
*/
|
||||||
#define XFS_RMAP_OFF_ATTR_FORK ((__uint64_t)1ULL << 63)
|
#define XFS_RMAP_OFF_ATTR_FORK ((uint64_t)1ULL << 63)
|
||||||
#define XFS_RMAP_OFF_BMBT_BLOCK ((__uint64_t)1ULL << 62)
|
#define XFS_RMAP_OFF_BMBT_BLOCK ((uint64_t)1ULL << 62)
|
||||||
#define XFS_RMAP_OFF_UNWRITTEN ((__uint64_t)1ULL << 61)
|
#define XFS_RMAP_OFF_UNWRITTEN ((uint64_t)1ULL << 61)
|
||||||
|
|
||||||
#define XFS_RMAP_LEN_MAX ((__uint32_t)~0U)
|
#define XFS_RMAP_LEN_MAX ((uint32_t)~0U)
|
||||||
#define XFS_RMAP_OFF_FLAGS (XFS_RMAP_OFF_ATTR_FORK | \
|
#define XFS_RMAP_OFF_FLAGS (XFS_RMAP_OFF_ATTR_FORK | \
|
||||||
XFS_RMAP_OFF_BMBT_BLOCK | \
|
XFS_RMAP_OFF_BMBT_BLOCK | \
|
||||||
XFS_RMAP_OFF_UNWRITTEN)
|
XFS_RMAP_OFF_UNWRITTEN)
|
||||||
#define XFS_RMAP_OFF_MASK ((__uint64_t)0x3FFFFFFFFFFFFFULL)
|
#define XFS_RMAP_OFF_MASK ((uint64_t)0x3FFFFFFFFFFFFFULL)
|
||||||
|
|
||||||
#define XFS_RMAP_OFF(off) ((off) & XFS_RMAP_OFF_MASK)
|
#define XFS_RMAP_OFF(off) ((off) & XFS_RMAP_OFF_MASK)
|
||||||
|
|
||||||
@ -1431,8 +1432,8 @@ struct xfs_rmap_rec {
|
|||||||
struct xfs_rmap_irec {
|
struct xfs_rmap_irec {
|
||||||
xfs_agblock_t rm_startblock; /* extent start block */
|
xfs_agblock_t rm_startblock; /* extent start block */
|
||||||
xfs_extlen_t rm_blockcount; /* extent length */
|
xfs_extlen_t rm_blockcount; /* extent length */
|
||||||
__uint64_t rm_owner; /* extent owner */
|
uint64_t rm_owner; /* extent owner */
|
||||||
__uint64_t rm_offset; /* offset within the owner */
|
uint64_t rm_offset; /* offset within the owner */
|
||||||
unsigned int rm_flags; /* state flags */
|
unsigned int rm_flags; /* state flags */
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -1544,11 +1545,11 @@ typedef struct xfs_bmbt_rec {
|
|||||||
__be64 l0, l1;
|
__be64 l0, l1;
|
||||||
} xfs_bmbt_rec_t;
|
} xfs_bmbt_rec_t;
|
||||||
|
|
||||||
typedef __uint64_t xfs_bmbt_rec_base_t; /* use this for casts */
|
typedef uint64_t xfs_bmbt_rec_base_t; /* use this for casts */
|
||||||
typedef xfs_bmbt_rec_t xfs_bmdr_rec_t;
|
typedef xfs_bmbt_rec_t xfs_bmdr_rec_t;
|
||||||
|
|
||||||
typedef struct xfs_bmbt_rec_host {
|
typedef struct xfs_bmbt_rec_host {
|
||||||
__uint64_t l0, l1;
|
uint64_t l0, l1;
|
||||||
} xfs_bmbt_rec_host_t;
|
} xfs_bmbt_rec_host_t;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -302,10 +302,10 @@ typedef struct xfs_bstat {
|
|||||||
* and using two 16bit values to hold new 32bit projid was choosen
|
* and using two 16bit values to hold new 32bit projid was choosen
|
||||||
* to retain compatibility with "old" filesystems).
|
* to retain compatibility with "old" filesystems).
|
||||||
*/
|
*/
|
||||||
static inline __uint32_t
|
static inline uint32_t
|
||||||
bstat_get_projid(struct xfs_bstat *bs)
|
bstat_get_projid(struct xfs_bstat *bs)
|
||||||
{
|
{
|
||||||
return (__uint32_t)bs->bs_projid_hi << 16 | bs->bs_projid_lo;
|
return (uint32_t)bs->bs_projid_hi << 16 | bs->bs_projid_lo;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -446,19 +446,15 @@ typedef struct xfs_handle {
|
|||||||
} xfs_handle_t;
|
} xfs_handle_t;
|
||||||
#define ha_fsid ha_u._ha_fsid
|
#define ha_fsid ha_u._ha_fsid
|
||||||
|
|
||||||
#define XFS_HSIZE(handle) (((char *) &(handle).ha_fid.fid_pad \
|
|
||||||
- (char *) &(handle)) \
|
|
||||||
+ (handle).ha_fid.fid_len)
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Structure passed to XFS_IOC_SWAPEXT
|
* Structure passed to XFS_IOC_SWAPEXT
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_swapext
|
typedef struct xfs_swapext
|
||||||
{
|
{
|
||||||
__int64_t sx_version; /* version */
|
int64_t sx_version; /* version */
|
||||||
#define XFS_SX_VERSION 0
|
#define XFS_SX_VERSION 0
|
||||||
__int64_t sx_fdtarget; /* fd of target file */
|
int64_t sx_fdtarget; /* fd of target file */
|
||||||
__int64_t sx_fdtmp; /* fd of tmp file */
|
int64_t sx_fdtmp; /* fd of tmp file */
|
||||||
xfs_off_t sx_offset; /* offset into file */
|
xfs_off_t sx_offset; /* offset into file */
|
||||||
xfs_off_t sx_length; /* leng from offset */
|
xfs_off_t sx_length; /* leng from offset */
|
||||||
char sx_pad[16]; /* pad space, unused */
|
char sx_pad[16]; /* pad space, unused */
|
||||||
@ -546,7 +542,7 @@ typedef struct xfs_swapext
|
|||||||
#define XFS_IOC_ATTRLIST_BY_HANDLE _IOW ('X', 122, struct xfs_fsop_attrlist_handlereq)
|
#define XFS_IOC_ATTRLIST_BY_HANDLE _IOW ('X', 122, struct xfs_fsop_attrlist_handlereq)
|
||||||
#define XFS_IOC_ATTRMULTI_BY_HANDLE _IOW ('X', 123, struct xfs_fsop_attrmulti_handlereq)
|
#define XFS_IOC_ATTRMULTI_BY_HANDLE _IOW ('X', 123, struct xfs_fsop_attrmulti_handlereq)
|
||||||
#define XFS_IOC_FSGEOMETRY _IOR ('X', 124, struct xfs_fsop_geom)
|
#define XFS_IOC_FSGEOMETRY _IOR ('X', 124, struct xfs_fsop_geom)
|
||||||
#define XFS_IOC_GOINGDOWN _IOR ('X', 125, __uint32_t)
|
#define XFS_IOC_GOINGDOWN _IOR ('X', 125, uint32_t)
|
||||||
/* XFS_IOC_GETFSUUID ---------- deprecated 140 */
|
/* XFS_IOC_GETFSUUID ---------- deprecated 140 */
|
||||||
|
|
||||||
|
|
||||||
|
@ -46,7 +46,7 @@
|
|||||||
/*
|
/*
|
||||||
* Allocation group level functions.
|
* Allocation group level functions.
|
||||||
*/
|
*/
|
||||||
static inline int
|
int
|
||||||
xfs_ialloc_cluster_alignment(
|
xfs_ialloc_cluster_alignment(
|
||||||
struct xfs_mount *mp)
|
struct xfs_mount *mp)
|
||||||
{
|
{
|
||||||
@ -98,24 +98,15 @@ xfs_inobt_update(
|
|||||||
return xfs_btree_update(cur, &rec);
|
return xfs_btree_update(cur, &rec);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/* Convert on-disk btree record to incore inobt record. */
|
||||||
* Get the data from the pointed-to record.
|
void
|
||||||
*/
|
xfs_inobt_btrec_to_irec(
|
||||||
int /* error */
|
struct xfs_mount *mp,
|
||||||
xfs_inobt_get_rec(
|
union xfs_btree_rec *rec,
|
||||||
struct xfs_btree_cur *cur, /* btree cursor */
|
struct xfs_inobt_rec_incore *irec)
|
||||||
xfs_inobt_rec_incore_t *irec, /* btree record */
|
|
||||||
int *stat) /* output: success/failure */
|
|
||||||
{
|
{
|
||||||
union xfs_btree_rec *rec;
|
|
||||||
int error;
|
|
||||||
|
|
||||||
error = xfs_btree_get_rec(cur, &rec, stat);
|
|
||||||
if (error || *stat == 0)
|
|
||||||
return error;
|
|
||||||
|
|
||||||
irec->ir_startino = be32_to_cpu(rec->inobt.ir_startino);
|
irec->ir_startino = be32_to_cpu(rec->inobt.ir_startino);
|
||||||
if (xfs_sb_version_hassparseinodes(&cur->bc_mp->m_sb)) {
|
if (xfs_sb_version_hassparseinodes(&mp->m_sb)) {
|
||||||
irec->ir_holemask = be16_to_cpu(rec->inobt.ir_u.sp.ir_holemask);
|
irec->ir_holemask = be16_to_cpu(rec->inobt.ir_u.sp.ir_holemask);
|
||||||
irec->ir_count = rec->inobt.ir_u.sp.ir_count;
|
irec->ir_count = rec->inobt.ir_u.sp.ir_count;
|
||||||
irec->ir_freecount = rec->inobt.ir_u.sp.ir_freecount;
|
irec->ir_freecount = rec->inobt.ir_u.sp.ir_freecount;
|
||||||
@ -130,6 +121,25 @@ xfs_inobt_get_rec(
|
|||||||
be32_to_cpu(rec->inobt.ir_u.f.ir_freecount);
|
be32_to_cpu(rec->inobt.ir_u.f.ir_freecount);
|
||||||
}
|
}
|
||||||
irec->ir_free = be64_to_cpu(rec->inobt.ir_free);
|
irec->ir_free = be64_to_cpu(rec->inobt.ir_free);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get the data from the pointed-to record.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
xfs_inobt_get_rec(
|
||||||
|
struct xfs_btree_cur *cur,
|
||||||
|
struct xfs_inobt_rec_incore *irec,
|
||||||
|
int *stat)
|
||||||
|
{
|
||||||
|
union xfs_btree_rec *rec;
|
||||||
|
int error;
|
||||||
|
|
||||||
|
error = xfs_btree_get_rec(cur, &rec, stat);
|
||||||
|
if (error || *stat == 0)
|
||||||
|
return error;
|
||||||
|
|
||||||
|
xfs_inobt_btrec_to_irec(cur->bc_mp, rec, irec);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -140,9 +150,9 @@ xfs_inobt_get_rec(
|
|||||||
STATIC int
|
STATIC int
|
||||||
xfs_inobt_insert_rec(
|
xfs_inobt_insert_rec(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
__uint16_t holemask,
|
uint16_t holemask,
|
||||||
__uint8_t count,
|
uint8_t count,
|
||||||
__int32_t freecount,
|
int32_t freecount,
|
||||||
xfs_inofree_t free,
|
xfs_inofree_t free,
|
||||||
int *stat)
|
int *stat)
|
||||||
{
|
{
|
||||||
@ -2542,8 +2552,7 @@ xfs_agi_read_verify(
|
|||||||
!xfs_buf_verify_cksum(bp, XFS_AGI_CRC_OFF))
|
!xfs_buf_verify_cksum(bp, XFS_AGI_CRC_OFF))
|
||||||
xfs_buf_ioerror(bp, -EFSBADCRC);
|
xfs_buf_ioerror(bp, -EFSBADCRC);
|
||||||
else if (XFS_TEST_ERROR(!xfs_agi_verify(bp), mp,
|
else if (XFS_TEST_ERROR(!xfs_agi_verify(bp), mp,
|
||||||
XFS_ERRTAG_IALLOC_READ_AGI,
|
XFS_ERRTAG_IALLOC_READ_AGI))
|
||||||
XFS_RANDOM_IALLOC_READ_AGI))
|
|
||||||
xfs_buf_ioerror(bp, -EFSCORRUPTED);
|
xfs_buf_ioerror(bp, -EFSCORRUPTED);
|
||||||
|
|
||||||
if (bp->b_error)
|
if (bp->b_error)
|
||||||
|
@ -168,5 +168,10 @@ int xfs_ialloc_inode_init(struct xfs_mount *mp, struct xfs_trans *tp,
|
|||||||
int xfs_read_agi(struct xfs_mount *mp, struct xfs_trans *tp,
|
int xfs_read_agi(struct xfs_mount *mp, struct xfs_trans *tp,
|
||||||
xfs_agnumber_t agno, struct xfs_buf **bpp);
|
xfs_agnumber_t agno, struct xfs_buf **bpp);
|
||||||
|
|
||||||
|
union xfs_btree_rec;
|
||||||
|
void xfs_inobt_btrec_to_irec(struct xfs_mount *mp, union xfs_btree_rec *rec,
|
||||||
|
struct xfs_inobt_rec_incore *irec);
|
||||||
|
|
||||||
|
int xfs_ialloc_cluster_alignment(struct xfs_mount *mp);
|
||||||
|
|
||||||
#endif /* __XFS_IALLOC_H__ */
|
#endif /* __XFS_IALLOC_H__ */
|
||||||
|
@ -174,6 +174,18 @@ xfs_inobt_init_key_from_rec(
|
|||||||
key->inobt.ir_startino = rec->inobt.ir_startino;
|
key->inobt.ir_startino = rec->inobt.ir_startino;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
STATIC void
|
||||||
|
xfs_inobt_init_high_key_from_rec(
|
||||||
|
union xfs_btree_key *key,
|
||||||
|
union xfs_btree_rec *rec)
|
||||||
|
{
|
||||||
|
__u32 x;
|
||||||
|
|
||||||
|
x = be32_to_cpu(rec->inobt.ir_startino);
|
||||||
|
x += XFS_INODES_PER_CHUNK - 1;
|
||||||
|
key->inobt.ir_startino = cpu_to_be32(x);
|
||||||
|
}
|
||||||
|
|
||||||
STATIC void
|
STATIC void
|
||||||
xfs_inobt_init_rec_from_cur(
|
xfs_inobt_init_rec_from_cur(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
@ -219,15 +231,25 @@ xfs_finobt_init_ptr_from_cur(
|
|||||||
ptr->s = agi->agi_free_root;
|
ptr->s = agi->agi_free_root;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_inobt_key_diff(
|
xfs_inobt_key_diff(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key)
|
union xfs_btree_key *key)
|
||||||
{
|
{
|
||||||
return (__int64_t)be32_to_cpu(key->inobt.ir_startino) -
|
return (int64_t)be32_to_cpu(key->inobt.ir_startino) -
|
||||||
cur->bc_rec.i.ir_startino;
|
cur->bc_rec.i.ir_startino;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
STATIC int64_t
|
||||||
|
xfs_inobt_diff_two_keys(
|
||||||
|
struct xfs_btree_cur *cur,
|
||||||
|
union xfs_btree_key *k1,
|
||||||
|
union xfs_btree_key *k2)
|
||||||
|
{
|
||||||
|
return (int64_t)be32_to_cpu(k1->inobt.ir_startino) -
|
||||||
|
be32_to_cpu(k2->inobt.ir_startino);
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
xfs_inobt_verify(
|
xfs_inobt_verify(
|
||||||
struct xfs_buf *bp)
|
struct xfs_buf *bp)
|
||||||
@ -302,7 +324,6 @@ const struct xfs_buf_ops xfs_inobt_buf_ops = {
|
|||||||
.verify_write = xfs_inobt_write_verify,
|
.verify_write = xfs_inobt_write_verify,
|
||||||
};
|
};
|
||||||
|
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
STATIC int
|
STATIC int
|
||||||
xfs_inobt_keys_inorder(
|
xfs_inobt_keys_inorder(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
@ -322,7 +343,6 @@ xfs_inobt_recs_inorder(
|
|||||||
return be32_to_cpu(r1->inobt.ir_startino) + XFS_INODES_PER_CHUNK <=
|
return be32_to_cpu(r1->inobt.ir_startino) + XFS_INODES_PER_CHUNK <=
|
||||||
be32_to_cpu(r2->inobt.ir_startino);
|
be32_to_cpu(r2->inobt.ir_startino);
|
||||||
}
|
}
|
||||||
#endif /* DEBUG */
|
|
||||||
|
|
||||||
static const struct xfs_btree_ops xfs_inobt_ops = {
|
static const struct xfs_btree_ops xfs_inobt_ops = {
|
||||||
.rec_len = sizeof(xfs_inobt_rec_t),
|
.rec_len = sizeof(xfs_inobt_rec_t),
|
||||||
@ -335,14 +355,14 @@ static const struct xfs_btree_ops xfs_inobt_ops = {
|
|||||||
.get_minrecs = xfs_inobt_get_minrecs,
|
.get_minrecs = xfs_inobt_get_minrecs,
|
||||||
.get_maxrecs = xfs_inobt_get_maxrecs,
|
.get_maxrecs = xfs_inobt_get_maxrecs,
|
||||||
.init_key_from_rec = xfs_inobt_init_key_from_rec,
|
.init_key_from_rec = xfs_inobt_init_key_from_rec,
|
||||||
|
.init_high_key_from_rec = xfs_inobt_init_high_key_from_rec,
|
||||||
.init_rec_from_cur = xfs_inobt_init_rec_from_cur,
|
.init_rec_from_cur = xfs_inobt_init_rec_from_cur,
|
||||||
.init_ptr_from_cur = xfs_inobt_init_ptr_from_cur,
|
.init_ptr_from_cur = xfs_inobt_init_ptr_from_cur,
|
||||||
.key_diff = xfs_inobt_key_diff,
|
.key_diff = xfs_inobt_key_diff,
|
||||||
.buf_ops = &xfs_inobt_buf_ops,
|
.buf_ops = &xfs_inobt_buf_ops,
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
.diff_two_keys = xfs_inobt_diff_two_keys,
|
||||||
.keys_inorder = xfs_inobt_keys_inorder,
|
.keys_inorder = xfs_inobt_keys_inorder,
|
||||||
.recs_inorder = xfs_inobt_recs_inorder,
|
.recs_inorder = xfs_inobt_recs_inorder,
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct xfs_btree_ops xfs_finobt_ops = {
|
static const struct xfs_btree_ops xfs_finobt_ops = {
|
||||||
@ -356,14 +376,14 @@ static const struct xfs_btree_ops xfs_finobt_ops = {
|
|||||||
.get_minrecs = xfs_inobt_get_minrecs,
|
.get_minrecs = xfs_inobt_get_minrecs,
|
||||||
.get_maxrecs = xfs_inobt_get_maxrecs,
|
.get_maxrecs = xfs_inobt_get_maxrecs,
|
||||||
.init_key_from_rec = xfs_inobt_init_key_from_rec,
|
.init_key_from_rec = xfs_inobt_init_key_from_rec,
|
||||||
|
.init_high_key_from_rec = xfs_inobt_init_high_key_from_rec,
|
||||||
.init_rec_from_cur = xfs_inobt_init_rec_from_cur,
|
.init_rec_from_cur = xfs_inobt_init_rec_from_cur,
|
||||||
.init_ptr_from_cur = xfs_finobt_init_ptr_from_cur,
|
.init_ptr_from_cur = xfs_finobt_init_ptr_from_cur,
|
||||||
.key_diff = xfs_inobt_key_diff,
|
.key_diff = xfs_inobt_key_diff,
|
||||||
.buf_ops = &xfs_inobt_buf_ops,
|
.buf_ops = &xfs_inobt_buf_ops,
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
.diff_two_keys = xfs_inobt_diff_two_keys,
|
||||||
.keys_inorder = xfs_inobt_keys_inorder,
|
.keys_inorder = xfs_inobt_keys_inorder,
|
||||||
.recs_inorder = xfs_inobt_recs_inorder,
|
.recs_inorder = xfs_inobt_recs_inorder,
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -105,8 +105,7 @@ xfs_inode_buf_verify(
|
|||||||
di_ok = dip->di_magic == cpu_to_be16(XFS_DINODE_MAGIC) &&
|
di_ok = dip->di_magic == cpu_to_be16(XFS_DINODE_MAGIC) &&
|
||||||
xfs_dinode_good_version(mp, dip->di_version);
|
xfs_dinode_good_version(mp, dip->di_version);
|
||||||
if (unlikely(XFS_TEST_ERROR(!di_ok, mp,
|
if (unlikely(XFS_TEST_ERROR(!di_ok, mp,
|
||||||
XFS_ERRTAG_ITOBP_INOTOBP,
|
XFS_ERRTAG_ITOBP_INOTOBP))) {
|
||||||
XFS_RANDOM_ITOBP_INOTOBP))) {
|
|
||||||
if (readahead) {
|
if (readahead) {
|
||||||
bp->b_flags &= ~XBF_DONE;
|
bp->b_flags &= ~XBF_DONE;
|
||||||
xfs_buf_ioerror(bp, -EIO);
|
xfs_buf_ioerror(bp, -EIO);
|
||||||
@ -381,7 +380,7 @@ xfs_log_dinode_to_disk(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
bool
|
||||||
xfs_dinode_verify(
|
xfs_dinode_verify(
|
||||||
struct xfs_mount *mp,
|
struct xfs_mount *mp,
|
||||||
xfs_ino_t ino,
|
xfs_ino_t ino,
|
||||||
@ -444,7 +443,7 @@ xfs_dinode_calc_crc(
|
|||||||
struct xfs_mount *mp,
|
struct xfs_mount *mp,
|
||||||
struct xfs_dinode *dip)
|
struct xfs_dinode *dip)
|
||||||
{
|
{
|
||||||
__uint32_t crc;
|
uint32_t crc;
|
||||||
|
|
||||||
if (dip->di_version < 3)
|
if (dip->di_version < 3)
|
||||||
return;
|
return;
|
||||||
|
@ -28,26 +28,26 @@ struct xfs_dinode;
|
|||||||
* format specific structures at the appropriate time.
|
* format specific structures at the appropriate time.
|
||||||
*/
|
*/
|
||||||
struct xfs_icdinode {
|
struct xfs_icdinode {
|
||||||
__int8_t di_version; /* inode version */
|
int8_t di_version; /* inode version */
|
||||||
__int8_t di_format; /* format of di_c data */
|
int8_t di_format; /* format of di_c data */
|
||||||
__uint16_t di_flushiter; /* incremented on flush */
|
uint16_t di_flushiter; /* incremented on flush */
|
||||||
__uint32_t di_uid; /* owner's user id */
|
uint32_t di_uid; /* owner's user id */
|
||||||
__uint32_t di_gid; /* owner's group id */
|
uint32_t di_gid; /* owner's group id */
|
||||||
__uint16_t di_projid_lo; /* lower part of owner's project id */
|
uint16_t di_projid_lo; /* lower part of owner's project id */
|
||||||
__uint16_t di_projid_hi; /* higher part of owner's project id */
|
uint16_t di_projid_hi; /* higher part of owner's project id */
|
||||||
xfs_fsize_t di_size; /* number of bytes in file */
|
xfs_fsize_t di_size; /* number of bytes in file */
|
||||||
xfs_rfsblock_t di_nblocks; /* # of direct & btree blocks used */
|
xfs_rfsblock_t di_nblocks; /* # of direct & btree blocks used */
|
||||||
xfs_extlen_t di_extsize; /* basic/minimum extent size for file */
|
xfs_extlen_t di_extsize; /* basic/minimum extent size for file */
|
||||||
xfs_extnum_t di_nextents; /* number of extents in data fork */
|
xfs_extnum_t di_nextents; /* number of extents in data fork */
|
||||||
xfs_aextnum_t di_anextents; /* number of extents in attribute fork*/
|
xfs_aextnum_t di_anextents; /* number of extents in attribute fork*/
|
||||||
__uint8_t di_forkoff; /* attr fork offs, <<3 for 64b align */
|
uint8_t di_forkoff; /* attr fork offs, <<3 for 64b align */
|
||||||
__int8_t di_aformat; /* format of attr fork's data */
|
int8_t di_aformat; /* format of attr fork's data */
|
||||||
__uint32_t di_dmevmask; /* DMIG event mask */
|
uint32_t di_dmevmask; /* DMIG event mask */
|
||||||
__uint16_t di_dmstate; /* DMIG state info */
|
uint16_t di_dmstate; /* DMIG state info */
|
||||||
__uint16_t di_flags; /* random flags, XFS_DIFLAG_... */
|
uint16_t di_flags; /* random flags, XFS_DIFLAG_... */
|
||||||
|
|
||||||
__uint64_t di_flags2; /* more random flags */
|
uint64_t di_flags2; /* more random flags */
|
||||||
__uint32_t di_cowextsize; /* basic cow extent size for file */
|
uint32_t di_cowextsize; /* basic cow extent size for file */
|
||||||
|
|
||||||
xfs_ictimestamp_t di_crtime; /* time created */
|
xfs_ictimestamp_t di_crtime; /* time created */
|
||||||
};
|
};
|
||||||
@ -82,4 +82,7 @@ void xfs_inobp_check(struct xfs_mount *, struct xfs_buf *);
|
|||||||
#define xfs_inobp_check(mp, bp)
|
#define xfs_inobp_check(mp, bp)
|
||||||
#endif /* DEBUG */
|
#endif /* DEBUG */
|
||||||
|
|
||||||
|
bool xfs_dinode_verify(struct xfs_mount *mp, xfs_ino_t ino,
|
||||||
|
struct xfs_dinode *dip);
|
||||||
|
|
||||||
#endif /* __XFS_INODE_BUF_H__ */
|
#endif /* __XFS_INODE_BUF_H__ */
|
||||||
|
@ -31,7 +31,7 @@ struct xfs_trans_res;
|
|||||||
* through all the log items definitions and everything they encode into the
|
* through all the log items definitions and everything they encode into the
|
||||||
* log.
|
* log.
|
||||||
*/
|
*/
|
||||||
typedef __uint32_t xlog_tid_t;
|
typedef uint32_t xlog_tid_t;
|
||||||
|
|
||||||
#define XLOG_MIN_ICLOGS 2
|
#define XLOG_MIN_ICLOGS 2
|
||||||
#define XLOG_MAX_ICLOGS 8
|
#define XLOG_MAX_ICLOGS 8
|
||||||
@ -211,7 +211,7 @@ typedef struct xfs_log_iovec {
|
|||||||
typedef struct xfs_trans_header {
|
typedef struct xfs_trans_header {
|
||||||
uint th_magic; /* magic number */
|
uint th_magic; /* magic number */
|
||||||
uint th_type; /* transaction type */
|
uint th_type; /* transaction type */
|
||||||
__int32_t th_tid; /* transaction id (unused) */
|
int32_t th_tid; /* transaction id (unused) */
|
||||||
uint th_num_items; /* num items logged by trans */
|
uint th_num_items; /* num items logged by trans */
|
||||||
} xfs_trans_header_t;
|
} xfs_trans_header_t;
|
||||||
|
|
||||||
@ -265,52 +265,52 @@ typedef struct xfs_trans_header {
|
|||||||
* must be added on to the end.
|
* must be added on to the end.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_inode_log_format {
|
typedef struct xfs_inode_log_format {
|
||||||
__uint16_t ilf_type; /* inode log item type */
|
uint16_t ilf_type; /* inode log item type */
|
||||||
__uint16_t ilf_size; /* size of this item */
|
uint16_t ilf_size; /* size of this item */
|
||||||
__uint32_t ilf_fields; /* flags for fields logged */
|
uint32_t ilf_fields; /* flags for fields logged */
|
||||||
__uint16_t ilf_asize; /* size of attr d/ext/root */
|
uint16_t ilf_asize; /* size of attr d/ext/root */
|
||||||
__uint16_t ilf_dsize; /* size of data/ext/root */
|
uint16_t ilf_dsize; /* size of data/ext/root */
|
||||||
__uint64_t ilf_ino; /* inode number */
|
uint64_t ilf_ino; /* inode number */
|
||||||
union {
|
union {
|
||||||
__uint32_t ilfu_rdev; /* rdev value for dev inode*/
|
uint32_t ilfu_rdev; /* rdev value for dev inode*/
|
||||||
uuid_t ilfu_uuid; /* mount point value */
|
uuid_t ilfu_uuid; /* mount point value */
|
||||||
} ilf_u;
|
} ilf_u;
|
||||||
__int64_t ilf_blkno; /* blkno of inode buffer */
|
int64_t ilf_blkno; /* blkno of inode buffer */
|
||||||
__int32_t ilf_len; /* len of inode buffer */
|
int32_t ilf_len; /* len of inode buffer */
|
||||||
__int32_t ilf_boffset; /* off of inode in buffer */
|
int32_t ilf_boffset; /* off of inode in buffer */
|
||||||
} xfs_inode_log_format_t;
|
} xfs_inode_log_format_t;
|
||||||
|
|
||||||
typedef struct xfs_inode_log_format_32 {
|
typedef struct xfs_inode_log_format_32 {
|
||||||
__uint16_t ilf_type; /* inode log item type */
|
uint16_t ilf_type; /* inode log item type */
|
||||||
__uint16_t ilf_size; /* size of this item */
|
uint16_t ilf_size; /* size of this item */
|
||||||
__uint32_t ilf_fields; /* flags for fields logged */
|
uint32_t ilf_fields; /* flags for fields logged */
|
||||||
__uint16_t ilf_asize; /* size of attr d/ext/root */
|
uint16_t ilf_asize; /* size of attr d/ext/root */
|
||||||
__uint16_t ilf_dsize; /* size of data/ext/root */
|
uint16_t ilf_dsize; /* size of data/ext/root */
|
||||||
__uint64_t ilf_ino; /* inode number */
|
uint64_t ilf_ino; /* inode number */
|
||||||
union {
|
union {
|
||||||
__uint32_t ilfu_rdev; /* rdev value for dev inode*/
|
uint32_t ilfu_rdev; /* rdev value for dev inode*/
|
||||||
uuid_t ilfu_uuid; /* mount point value */
|
uuid_t ilfu_uuid; /* mount point value */
|
||||||
} ilf_u;
|
} ilf_u;
|
||||||
__int64_t ilf_blkno; /* blkno of inode buffer */
|
int64_t ilf_blkno; /* blkno of inode buffer */
|
||||||
__int32_t ilf_len; /* len of inode buffer */
|
int32_t ilf_len; /* len of inode buffer */
|
||||||
__int32_t ilf_boffset; /* off of inode in buffer */
|
int32_t ilf_boffset; /* off of inode in buffer */
|
||||||
} __attribute__((packed)) xfs_inode_log_format_32_t;
|
} __attribute__((packed)) xfs_inode_log_format_32_t;
|
||||||
|
|
||||||
typedef struct xfs_inode_log_format_64 {
|
typedef struct xfs_inode_log_format_64 {
|
||||||
__uint16_t ilf_type; /* inode log item type */
|
uint16_t ilf_type; /* inode log item type */
|
||||||
__uint16_t ilf_size; /* size of this item */
|
uint16_t ilf_size; /* size of this item */
|
||||||
__uint32_t ilf_fields; /* flags for fields logged */
|
uint32_t ilf_fields; /* flags for fields logged */
|
||||||
__uint16_t ilf_asize; /* size of attr d/ext/root */
|
uint16_t ilf_asize; /* size of attr d/ext/root */
|
||||||
__uint16_t ilf_dsize; /* size of data/ext/root */
|
uint16_t ilf_dsize; /* size of data/ext/root */
|
||||||
__uint32_t ilf_pad; /* pad for 64 bit boundary */
|
uint32_t ilf_pad; /* pad for 64 bit boundary */
|
||||||
__uint64_t ilf_ino; /* inode number */
|
uint64_t ilf_ino; /* inode number */
|
||||||
union {
|
union {
|
||||||
__uint32_t ilfu_rdev; /* rdev value for dev inode*/
|
uint32_t ilfu_rdev; /* rdev value for dev inode*/
|
||||||
uuid_t ilfu_uuid; /* mount point value */
|
uuid_t ilfu_uuid; /* mount point value */
|
||||||
} ilf_u;
|
} ilf_u;
|
||||||
__int64_t ilf_blkno; /* blkno of inode buffer */
|
int64_t ilf_blkno; /* blkno of inode buffer */
|
||||||
__int32_t ilf_len; /* len of inode buffer */
|
int32_t ilf_len; /* len of inode buffer */
|
||||||
__int32_t ilf_boffset; /* off of inode in buffer */
|
int32_t ilf_boffset; /* off of inode in buffer */
|
||||||
} xfs_inode_log_format_64_t;
|
} xfs_inode_log_format_64_t;
|
||||||
|
|
||||||
|
|
||||||
@ -379,8 +379,8 @@ static inline int xfs_ilog_fdata(int w)
|
|||||||
* information.
|
* information.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_ictimestamp {
|
typedef struct xfs_ictimestamp {
|
||||||
__int32_t t_sec; /* timestamp seconds */
|
int32_t t_sec; /* timestamp seconds */
|
||||||
__int32_t t_nsec; /* timestamp nanoseconds */
|
int32_t t_nsec; /* timestamp nanoseconds */
|
||||||
} xfs_ictimestamp_t;
|
} xfs_ictimestamp_t;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -388,18 +388,18 @@ typedef struct xfs_ictimestamp {
|
|||||||
* kept identical to struct xfs_dinode except for the endianness annotations.
|
* kept identical to struct xfs_dinode except for the endianness annotations.
|
||||||
*/
|
*/
|
||||||
struct xfs_log_dinode {
|
struct xfs_log_dinode {
|
||||||
__uint16_t di_magic; /* inode magic # = XFS_DINODE_MAGIC */
|
uint16_t di_magic; /* inode magic # = XFS_DINODE_MAGIC */
|
||||||
__uint16_t di_mode; /* mode and type of file */
|
uint16_t di_mode; /* mode and type of file */
|
||||||
__int8_t di_version; /* inode version */
|
int8_t di_version; /* inode version */
|
||||||
__int8_t di_format; /* format of di_c data */
|
int8_t di_format; /* format of di_c data */
|
||||||
__uint8_t di_pad3[2]; /* unused in v2/3 inodes */
|
uint8_t di_pad3[2]; /* unused in v2/3 inodes */
|
||||||
__uint32_t di_uid; /* owner's user id */
|
uint32_t di_uid; /* owner's user id */
|
||||||
__uint32_t di_gid; /* owner's group id */
|
uint32_t di_gid; /* owner's group id */
|
||||||
__uint32_t di_nlink; /* number of links to file */
|
uint32_t di_nlink; /* number of links to file */
|
||||||
__uint16_t di_projid_lo; /* lower part of owner's project id */
|
uint16_t di_projid_lo; /* lower part of owner's project id */
|
||||||
__uint16_t di_projid_hi; /* higher part of owner's project id */
|
uint16_t di_projid_hi; /* higher part of owner's project id */
|
||||||
__uint8_t di_pad[6]; /* unused, zeroed space */
|
uint8_t di_pad[6]; /* unused, zeroed space */
|
||||||
__uint16_t di_flushiter; /* incremented on flush */
|
uint16_t di_flushiter; /* incremented on flush */
|
||||||
xfs_ictimestamp_t di_atime; /* time last accessed */
|
xfs_ictimestamp_t di_atime; /* time last accessed */
|
||||||
xfs_ictimestamp_t di_mtime; /* time last modified */
|
xfs_ictimestamp_t di_mtime; /* time last modified */
|
||||||
xfs_ictimestamp_t di_ctime; /* time created/inode modified */
|
xfs_ictimestamp_t di_ctime; /* time created/inode modified */
|
||||||
@ -408,23 +408,23 @@ struct xfs_log_dinode {
|
|||||||
xfs_extlen_t di_extsize; /* basic/minimum extent size for file */
|
xfs_extlen_t di_extsize; /* basic/minimum extent size for file */
|
||||||
xfs_extnum_t di_nextents; /* number of extents in data fork */
|
xfs_extnum_t di_nextents; /* number of extents in data fork */
|
||||||
xfs_aextnum_t di_anextents; /* number of extents in attribute fork*/
|
xfs_aextnum_t di_anextents; /* number of extents in attribute fork*/
|
||||||
__uint8_t di_forkoff; /* attr fork offs, <<3 for 64b align */
|
uint8_t di_forkoff; /* attr fork offs, <<3 for 64b align */
|
||||||
__int8_t di_aformat; /* format of attr fork's data */
|
int8_t di_aformat; /* format of attr fork's data */
|
||||||
__uint32_t di_dmevmask; /* DMIG event mask */
|
uint32_t di_dmevmask; /* DMIG event mask */
|
||||||
__uint16_t di_dmstate; /* DMIG state info */
|
uint16_t di_dmstate; /* DMIG state info */
|
||||||
__uint16_t di_flags; /* random flags, XFS_DIFLAG_... */
|
uint16_t di_flags; /* random flags, XFS_DIFLAG_... */
|
||||||
__uint32_t di_gen; /* generation number */
|
uint32_t di_gen; /* generation number */
|
||||||
|
|
||||||
/* di_next_unlinked is the only non-core field in the old dinode */
|
/* di_next_unlinked is the only non-core field in the old dinode */
|
||||||
xfs_agino_t di_next_unlinked;/* agi unlinked list ptr */
|
xfs_agino_t di_next_unlinked;/* agi unlinked list ptr */
|
||||||
|
|
||||||
/* start of the extended dinode, writable fields */
|
/* start of the extended dinode, writable fields */
|
||||||
__uint32_t di_crc; /* CRC of the inode */
|
uint32_t di_crc; /* CRC of the inode */
|
||||||
__uint64_t di_changecount; /* number of attribute changes */
|
uint64_t di_changecount; /* number of attribute changes */
|
||||||
xfs_lsn_t di_lsn; /* flush sequence */
|
xfs_lsn_t di_lsn; /* flush sequence */
|
||||||
__uint64_t di_flags2; /* more random flags */
|
uint64_t di_flags2; /* more random flags */
|
||||||
__uint32_t di_cowextsize; /* basic cow extent size for file */
|
uint32_t di_cowextsize; /* basic cow extent size for file */
|
||||||
__uint8_t di_pad2[12]; /* more padding for future expansion */
|
uint8_t di_pad2[12]; /* more padding for future expansion */
|
||||||
|
|
||||||
/* fields only written to during inode creation */
|
/* fields only written to during inode creation */
|
||||||
xfs_ictimestamp_t di_crtime; /* time created */
|
xfs_ictimestamp_t di_crtime; /* time created */
|
||||||
@ -483,7 +483,7 @@ typedef struct xfs_buf_log_format {
|
|||||||
unsigned short blf_size; /* size of this item */
|
unsigned short blf_size; /* size of this item */
|
||||||
unsigned short blf_flags; /* misc state */
|
unsigned short blf_flags; /* misc state */
|
||||||
unsigned short blf_len; /* number of blocks in this buf */
|
unsigned short blf_len; /* number of blocks in this buf */
|
||||||
__int64_t blf_blkno; /* starting blkno of this buf */
|
int64_t blf_blkno; /* starting blkno of this buf */
|
||||||
unsigned int blf_map_size; /* used size of data bitmap in words */
|
unsigned int blf_map_size; /* used size of data bitmap in words */
|
||||||
unsigned int blf_data_map[XFS_BLF_DATAMAP_SIZE]; /* dirty bitmap */
|
unsigned int blf_data_map[XFS_BLF_DATAMAP_SIZE]; /* dirty bitmap */
|
||||||
} xfs_buf_log_format_t;
|
} xfs_buf_log_format_t;
|
||||||
@ -533,7 +533,7 @@ xfs_blft_to_flags(struct xfs_buf_log_format *blf, enum xfs_blft type)
|
|||||||
blf->blf_flags |= ((type << XFS_BLFT_SHIFT) & XFS_BLFT_MASK);
|
blf->blf_flags |= ((type << XFS_BLFT_SHIFT) & XFS_BLFT_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline __uint16_t
|
static inline uint16_t
|
||||||
xfs_blft_from_flags(struct xfs_buf_log_format *blf)
|
xfs_blft_from_flags(struct xfs_buf_log_format *blf)
|
||||||
{
|
{
|
||||||
return (blf->blf_flags & XFS_BLFT_MASK) >> XFS_BLFT_SHIFT;
|
return (blf->blf_flags & XFS_BLFT_MASK) >> XFS_BLFT_SHIFT;
|
||||||
@ -554,14 +554,14 @@ typedef struct xfs_extent {
|
|||||||
* conversion routine.
|
* conversion routine.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_extent_32 {
|
typedef struct xfs_extent_32 {
|
||||||
__uint64_t ext_start;
|
uint64_t ext_start;
|
||||||
__uint32_t ext_len;
|
uint32_t ext_len;
|
||||||
} __attribute__((packed)) xfs_extent_32_t;
|
} __attribute__((packed)) xfs_extent_32_t;
|
||||||
|
|
||||||
typedef struct xfs_extent_64 {
|
typedef struct xfs_extent_64 {
|
||||||
__uint64_t ext_start;
|
uint64_t ext_start;
|
||||||
__uint32_t ext_len;
|
uint32_t ext_len;
|
||||||
__uint32_t ext_pad;
|
uint32_t ext_pad;
|
||||||
} xfs_extent_64_t;
|
} xfs_extent_64_t;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -570,26 +570,26 @@ typedef struct xfs_extent_64 {
|
|||||||
* size is given by efi_nextents.
|
* size is given by efi_nextents.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_efi_log_format {
|
typedef struct xfs_efi_log_format {
|
||||||
__uint16_t efi_type; /* efi log item type */
|
uint16_t efi_type; /* efi log item type */
|
||||||
__uint16_t efi_size; /* size of this item */
|
uint16_t efi_size; /* size of this item */
|
||||||
__uint32_t efi_nextents; /* # extents to free */
|
uint32_t efi_nextents; /* # extents to free */
|
||||||
__uint64_t efi_id; /* efi identifier */
|
uint64_t efi_id; /* efi identifier */
|
||||||
xfs_extent_t efi_extents[1]; /* array of extents to free */
|
xfs_extent_t efi_extents[1]; /* array of extents to free */
|
||||||
} xfs_efi_log_format_t;
|
} xfs_efi_log_format_t;
|
||||||
|
|
||||||
typedef struct xfs_efi_log_format_32 {
|
typedef struct xfs_efi_log_format_32 {
|
||||||
__uint16_t efi_type; /* efi log item type */
|
uint16_t efi_type; /* efi log item type */
|
||||||
__uint16_t efi_size; /* size of this item */
|
uint16_t efi_size; /* size of this item */
|
||||||
__uint32_t efi_nextents; /* # extents to free */
|
uint32_t efi_nextents; /* # extents to free */
|
||||||
__uint64_t efi_id; /* efi identifier */
|
uint64_t efi_id; /* efi identifier */
|
||||||
xfs_extent_32_t efi_extents[1]; /* array of extents to free */
|
xfs_extent_32_t efi_extents[1]; /* array of extents to free */
|
||||||
} __attribute__((packed)) xfs_efi_log_format_32_t;
|
} __attribute__((packed)) xfs_efi_log_format_32_t;
|
||||||
|
|
||||||
typedef struct xfs_efi_log_format_64 {
|
typedef struct xfs_efi_log_format_64 {
|
||||||
__uint16_t efi_type; /* efi log item type */
|
uint16_t efi_type; /* efi log item type */
|
||||||
__uint16_t efi_size; /* size of this item */
|
uint16_t efi_size; /* size of this item */
|
||||||
__uint32_t efi_nextents; /* # extents to free */
|
uint32_t efi_nextents; /* # extents to free */
|
||||||
__uint64_t efi_id; /* efi identifier */
|
uint64_t efi_id; /* efi identifier */
|
||||||
xfs_extent_64_t efi_extents[1]; /* array of extents to free */
|
xfs_extent_64_t efi_extents[1]; /* array of extents to free */
|
||||||
} xfs_efi_log_format_64_t;
|
} xfs_efi_log_format_64_t;
|
||||||
|
|
||||||
@ -599,26 +599,26 @@ typedef struct xfs_efi_log_format_64 {
|
|||||||
* size is given by efd_nextents;
|
* size is given by efd_nextents;
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_efd_log_format {
|
typedef struct xfs_efd_log_format {
|
||||||
__uint16_t efd_type; /* efd log item type */
|
uint16_t efd_type; /* efd log item type */
|
||||||
__uint16_t efd_size; /* size of this item */
|
uint16_t efd_size; /* size of this item */
|
||||||
__uint32_t efd_nextents; /* # of extents freed */
|
uint32_t efd_nextents; /* # of extents freed */
|
||||||
__uint64_t efd_efi_id; /* id of corresponding efi */
|
uint64_t efd_efi_id; /* id of corresponding efi */
|
||||||
xfs_extent_t efd_extents[1]; /* array of extents freed */
|
xfs_extent_t efd_extents[1]; /* array of extents freed */
|
||||||
} xfs_efd_log_format_t;
|
} xfs_efd_log_format_t;
|
||||||
|
|
||||||
typedef struct xfs_efd_log_format_32 {
|
typedef struct xfs_efd_log_format_32 {
|
||||||
__uint16_t efd_type; /* efd log item type */
|
uint16_t efd_type; /* efd log item type */
|
||||||
__uint16_t efd_size; /* size of this item */
|
uint16_t efd_size; /* size of this item */
|
||||||
__uint32_t efd_nextents; /* # of extents freed */
|
uint32_t efd_nextents; /* # of extents freed */
|
||||||
__uint64_t efd_efi_id; /* id of corresponding efi */
|
uint64_t efd_efi_id; /* id of corresponding efi */
|
||||||
xfs_extent_32_t efd_extents[1]; /* array of extents freed */
|
xfs_extent_32_t efd_extents[1]; /* array of extents freed */
|
||||||
} __attribute__((packed)) xfs_efd_log_format_32_t;
|
} __attribute__((packed)) xfs_efd_log_format_32_t;
|
||||||
|
|
||||||
typedef struct xfs_efd_log_format_64 {
|
typedef struct xfs_efd_log_format_64 {
|
||||||
__uint16_t efd_type; /* efd log item type */
|
uint16_t efd_type; /* efd log item type */
|
||||||
__uint16_t efd_size; /* size of this item */
|
uint16_t efd_size; /* size of this item */
|
||||||
__uint32_t efd_nextents; /* # of extents freed */
|
uint32_t efd_nextents; /* # of extents freed */
|
||||||
__uint64_t efd_efi_id; /* id of corresponding efi */
|
uint64_t efd_efi_id; /* id of corresponding efi */
|
||||||
xfs_extent_64_t efd_extents[1]; /* array of extents freed */
|
xfs_extent_64_t efd_extents[1]; /* array of extents freed */
|
||||||
} xfs_efd_log_format_64_t;
|
} xfs_efd_log_format_64_t;
|
||||||
|
|
||||||
@ -626,11 +626,11 @@ typedef struct xfs_efd_log_format_64 {
|
|||||||
* RUI/RUD (reverse mapping) log format definitions
|
* RUI/RUD (reverse mapping) log format definitions
|
||||||
*/
|
*/
|
||||||
struct xfs_map_extent {
|
struct xfs_map_extent {
|
||||||
__uint64_t me_owner;
|
uint64_t me_owner;
|
||||||
__uint64_t me_startblock;
|
uint64_t me_startblock;
|
||||||
__uint64_t me_startoff;
|
uint64_t me_startoff;
|
||||||
__uint32_t me_len;
|
uint32_t me_len;
|
||||||
__uint32_t me_flags;
|
uint32_t me_flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* rmap me_flags: upper bits are flags, lower byte is type code */
|
/* rmap me_flags: upper bits are flags, lower byte is type code */
|
||||||
@ -659,10 +659,10 @@ struct xfs_map_extent {
|
|||||||
* size is given by rui_nextents.
|
* size is given by rui_nextents.
|
||||||
*/
|
*/
|
||||||
struct xfs_rui_log_format {
|
struct xfs_rui_log_format {
|
||||||
__uint16_t rui_type; /* rui log item type */
|
uint16_t rui_type; /* rui log item type */
|
||||||
__uint16_t rui_size; /* size of this item */
|
uint16_t rui_size; /* size of this item */
|
||||||
__uint32_t rui_nextents; /* # extents to free */
|
uint32_t rui_nextents; /* # extents to free */
|
||||||
__uint64_t rui_id; /* rui identifier */
|
uint64_t rui_id; /* rui identifier */
|
||||||
struct xfs_map_extent rui_extents[]; /* array of extents to rmap */
|
struct xfs_map_extent rui_extents[]; /* array of extents to rmap */
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -680,19 +680,19 @@ xfs_rui_log_format_sizeof(
|
|||||||
* size is given by rud_nextents;
|
* size is given by rud_nextents;
|
||||||
*/
|
*/
|
||||||
struct xfs_rud_log_format {
|
struct xfs_rud_log_format {
|
||||||
__uint16_t rud_type; /* rud log item type */
|
uint16_t rud_type; /* rud log item type */
|
||||||
__uint16_t rud_size; /* size of this item */
|
uint16_t rud_size; /* size of this item */
|
||||||
__uint32_t __pad;
|
uint32_t __pad;
|
||||||
__uint64_t rud_rui_id; /* id of corresponding rui */
|
uint64_t rud_rui_id; /* id of corresponding rui */
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* CUI/CUD (refcount update) log format definitions
|
* CUI/CUD (refcount update) log format definitions
|
||||||
*/
|
*/
|
||||||
struct xfs_phys_extent {
|
struct xfs_phys_extent {
|
||||||
__uint64_t pe_startblock;
|
uint64_t pe_startblock;
|
||||||
__uint32_t pe_len;
|
uint32_t pe_len;
|
||||||
__uint32_t pe_flags;
|
uint32_t pe_flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* refcount pe_flags: upper bits are flags, lower byte is type code */
|
/* refcount pe_flags: upper bits are flags, lower byte is type code */
|
||||||
@ -707,10 +707,10 @@ struct xfs_phys_extent {
|
|||||||
* size is given by cui_nextents.
|
* size is given by cui_nextents.
|
||||||
*/
|
*/
|
||||||
struct xfs_cui_log_format {
|
struct xfs_cui_log_format {
|
||||||
__uint16_t cui_type; /* cui log item type */
|
uint16_t cui_type; /* cui log item type */
|
||||||
__uint16_t cui_size; /* size of this item */
|
uint16_t cui_size; /* size of this item */
|
||||||
__uint32_t cui_nextents; /* # extents to free */
|
uint32_t cui_nextents; /* # extents to free */
|
||||||
__uint64_t cui_id; /* cui identifier */
|
uint64_t cui_id; /* cui identifier */
|
||||||
struct xfs_phys_extent cui_extents[]; /* array of extents */
|
struct xfs_phys_extent cui_extents[]; /* array of extents */
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -728,10 +728,10 @@ xfs_cui_log_format_sizeof(
|
|||||||
* size is given by cud_nextents;
|
* size is given by cud_nextents;
|
||||||
*/
|
*/
|
||||||
struct xfs_cud_log_format {
|
struct xfs_cud_log_format {
|
||||||
__uint16_t cud_type; /* cud log item type */
|
uint16_t cud_type; /* cud log item type */
|
||||||
__uint16_t cud_size; /* size of this item */
|
uint16_t cud_size; /* size of this item */
|
||||||
__uint32_t __pad;
|
uint32_t __pad;
|
||||||
__uint64_t cud_cui_id; /* id of corresponding cui */
|
uint64_t cud_cui_id; /* id of corresponding cui */
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -755,10 +755,10 @@ struct xfs_cud_log_format {
|
|||||||
* size is given by bui_nextents.
|
* size is given by bui_nextents.
|
||||||
*/
|
*/
|
||||||
struct xfs_bui_log_format {
|
struct xfs_bui_log_format {
|
||||||
__uint16_t bui_type; /* bui log item type */
|
uint16_t bui_type; /* bui log item type */
|
||||||
__uint16_t bui_size; /* size of this item */
|
uint16_t bui_size; /* size of this item */
|
||||||
__uint32_t bui_nextents; /* # extents to free */
|
uint32_t bui_nextents; /* # extents to free */
|
||||||
__uint64_t bui_id; /* bui identifier */
|
uint64_t bui_id; /* bui identifier */
|
||||||
struct xfs_map_extent bui_extents[]; /* array of extents to bmap */
|
struct xfs_map_extent bui_extents[]; /* array of extents to bmap */
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -776,10 +776,10 @@ xfs_bui_log_format_sizeof(
|
|||||||
* size is given by bud_nextents;
|
* size is given by bud_nextents;
|
||||||
*/
|
*/
|
||||||
struct xfs_bud_log_format {
|
struct xfs_bud_log_format {
|
||||||
__uint16_t bud_type; /* bud log item type */
|
uint16_t bud_type; /* bud log item type */
|
||||||
__uint16_t bud_size; /* size of this item */
|
uint16_t bud_size; /* size of this item */
|
||||||
__uint32_t __pad;
|
uint32_t __pad;
|
||||||
__uint64_t bud_bui_id; /* id of corresponding bui */
|
uint64_t bud_bui_id; /* id of corresponding bui */
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -789,12 +789,12 @@ struct xfs_bud_log_format {
|
|||||||
* 32 bits : log_recovery code assumes that.
|
* 32 bits : log_recovery code assumes that.
|
||||||
*/
|
*/
|
||||||
typedef struct xfs_dq_logformat {
|
typedef struct xfs_dq_logformat {
|
||||||
__uint16_t qlf_type; /* dquot log item type */
|
uint16_t qlf_type; /* dquot log item type */
|
||||||
__uint16_t qlf_size; /* size of this item */
|
uint16_t qlf_size; /* size of this item */
|
||||||
xfs_dqid_t qlf_id; /* usr/grp/proj id : 32 bits */
|
xfs_dqid_t qlf_id; /* usr/grp/proj id : 32 bits */
|
||||||
__int64_t qlf_blkno; /* blkno of dquot buffer */
|
int64_t qlf_blkno; /* blkno of dquot buffer */
|
||||||
__int32_t qlf_len; /* len of dquot buffer */
|
int32_t qlf_len; /* len of dquot buffer */
|
||||||
__uint32_t qlf_boffset; /* off of dquot in buffer */
|
uint32_t qlf_boffset; /* off of dquot in buffer */
|
||||||
} xfs_dq_logformat_t;
|
} xfs_dq_logformat_t;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -853,8 +853,8 @@ typedef struct xfs_qoff_logformat {
|
|||||||
* decoding can be done correctly.
|
* decoding can be done correctly.
|
||||||
*/
|
*/
|
||||||
struct xfs_icreate_log {
|
struct xfs_icreate_log {
|
||||||
__uint16_t icl_type; /* type of log format structure */
|
uint16_t icl_type; /* type of log format structure */
|
||||||
__uint16_t icl_size; /* size of log format structure */
|
uint16_t icl_size; /* size of log format structure */
|
||||||
__be32 icl_ag; /* ag being allocated in */
|
__be32 icl_ag; /* ag being allocated in */
|
||||||
__be32 icl_agbno; /* start block of inode range */
|
__be32 icl_agbno; /* start block of inode range */
|
||||||
__be32 icl_count; /* number of inodes to initialise */
|
__be32 icl_count; /* number of inodes to initialise */
|
||||||
|
@ -26,7 +26,7 @@
|
|||||||
#define XLOG_RHASH_SIZE 16
|
#define XLOG_RHASH_SIZE 16
|
||||||
#define XLOG_RHASH_SHIFT 2
|
#define XLOG_RHASH_SHIFT 2
|
||||||
#define XLOG_RHASH(tid) \
|
#define XLOG_RHASH(tid) \
|
||||||
((((__uint32_t)tid)>>XLOG_RHASH_SHIFT) & (XLOG_RHASH_SIZE-1))
|
((((uint32_t)tid)>>XLOG_RHASH_SHIFT) & (XLOG_RHASH_SIZE-1))
|
||||||
|
|
||||||
#define XLOG_MAX_REGIONS_IN_ITEM (XFS_MAX_BLOCKSIZE / XFS_BLF_CHUNK / 2 + 1)
|
#define XLOG_MAX_REGIONS_IN_ITEM (XFS_MAX_BLOCKSIZE / XFS_BLF_CHUNK / 2 + 1)
|
||||||
|
|
||||||
|
@ -27,8 +27,8 @@
|
|||||||
* they may need 64-bit accounting. Hence, 64-bit quota-counters,
|
* they may need 64-bit accounting. Hence, 64-bit quota-counters,
|
||||||
* and quota-limits. This is a waste in the common case, but hey ...
|
* and quota-limits. This is a waste in the common case, but hey ...
|
||||||
*/
|
*/
|
||||||
typedef __uint64_t xfs_qcnt_t;
|
typedef uint64_t xfs_qcnt_t;
|
||||||
typedef __uint16_t xfs_qwarncnt_t;
|
typedef uint16_t xfs_qwarncnt_t;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* flags for q_flags field in the dquot.
|
* flags for q_flags field in the dquot.
|
||||||
@ -136,6 +136,8 @@ typedef __uint16_t xfs_qwarncnt_t;
|
|||||||
*/
|
*/
|
||||||
#define XFS_QMOPT_INHERIT 0x1000000
|
#define XFS_QMOPT_INHERIT 0x1000000
|
||||||
|
|
||||||
|
#define XFS_QMOPT_NOLOCK 0x2000000 /* don't ilock during dqget */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* flags to xfs_trans_mod_dquot.
|
* flags to xfs_trans_mod_dquot.
|
||||||
*/
|
*/
|
||||||
|
@ -784,14 +784,6 @@ xfs_refcount_merge_extents(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* While we're adjusting the refcounts records of an extent, we have
|
|
||||||
* to keep an eye on the number of extents we're dirtying -- run too
|
|
||||||
* many in a single transaction and we'll exceed the transaction's
|
|
||||||
* reservation and crash the fs. Each record adds 12 bytes to the
|
|
||||||
* log (plus any key updates) so we'll conservatively assume 24 bytes
|
|
||||||
* per record. We must also leave space for btree splits on both ends
|
|
||||||
* of the range and space for the CUD and a new CUI.
|
|
||||||
*
|
|
||||||
* XXX: This is a pretty hand-wavy estimate. The penalty for guessing
|
* XXX: This is a pretty hand-wavy estimate. The penalty for guessing
|
||||||
* true incorrectly is a shutdown FS; the penalty for guessing false
|
* true incorrectly is a shutdown FS; the penalty for guessing false
|
||||||
* incorrectly is more transaction rolls than might be necessary.
|
* incorrectly is more transaction rolls than might be necessary.
|
||||||
@ -813,8 +805,7 @@ xfs_refcount_still_have_space(
|
|||||||
*/
|
*/
|
||||||
if (cur->bc_private.a.priv.refc.nr_ops > 2 &&
|
if (cur->bc_private.a.priv.refc.nr_ops > 2 &&
|
||||||
XFS_TEST_ERROR(false, cur->bc_mp,
|
XFS_TEST_ERROR(false, cur->bc_mp,
|
||||||
XFS_ERRTAG_REFCOUNT_CONTINUE_UPDATE,
|
XFS_ERRTAG_REFCOUNT_CONTINUE_UPDATE))
|
||||||
XFS_RANDOM_REFCOUNT_CONTINUE_UPDATE))
|
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (cur->bc_private.a.priv.refc.nr_ops == 0)
|
if (cur->bc_private.a.priv.refc.nr_ops == 0)
|
||||||
@ -822,7 +813,7 @@ xfs_refcount_still_have_space(
|
|||||||
else if (overhead > cur->bc_tp->t_log_res)
|
else if (overhead > cur->bc_tp->t_log_res)
|
||||||
return false;
|
return false;
|
||||||
return cur->bc_tp->t_log_res - overhead >
|
return cur->bc_tp->t_log_res - overhead >
|
||||||
cur->bc_private.a.priv.refc.nr_ops * 32;
|
cur->bc_private.a.priv.refc.nr_ops * XFS_REFCOUNT_ITEM_OVERHEAD;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1076,8 +1067,7 @@ xfs_refcount_finish_one(
|
|||||||
blockcount);
|
blockcount);
|
||||||
|
|
||||||
if (XFS_TEST_ERROR(false, mp,
|
if (XFS_TEST_ERROR(false, mp,
|
||||||
XFS_ERRTAG_REFCOUNT_FINISH_ONE,
|
XFS_ERRTAG_REFCOUNT_FINISH_ONE))
|
||||||
XFS_RANDOM_REFCOUNT_FINISH_ONE))
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -67,4 +67,20 @@ extern int xfs_refcount_free_cow_extent(struct xfs_mount *mp,
|
|||||||
extern int xfs_refcount_recover_cow_leftovers(struct xfs_mount *mp,
|
extern int xfs_refcount_recover_cow_leftovers(struct xfs_mount *mp,
|
||||||
xfs_agnumber_t agno);
|
xfs_agnumber_t agno);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* While we're adjusting the refcounts records of an extent, we have
|
||||||
|
* to keep an eye on the number of extents we're dirtying -- run too
|
||||||
|
* many in a single transaction and we'll exceed the transaction's
|
||||||
|
* reservation and crash the fs. Each record adds 12 bytes to the
|
||||||
|
* log (plus any key updates) so we'll conservatively assume 32 bytes
|
||||||
|
* per record. We must also leave space for btree splits on both ends
|
||||||
|
* of the range and space for the CUD and a new CUI.
|
||||||
|
*/
|
||||||
|
#define XFS_REFCOUNT_ITEM_OVERHEAD 32
|
||||||
|
|
||||||
|
static inline xfs_fileoff_t xfs_refcount_max_unmap(int log_res)
|
||||||
|
{
|
||||||
|
return (log_res * 3 / 4) / XFS_REFCOUNT_ITEM_OVERHEAD;
|
||||||
|
}
|
||||||
|
|
||||||
#endif /* __XFS_REFCOUNT_H__ */
|
#endif /* __XFS_REFCOUNT_H__ */
|
||||||
|
@ -202,7 +202,7 @@ xfs_refcountbt_init_ptr_from_cur(
|
|||||||
ptr->s = agf->agf_refcount_root;
|
ptr->s = agf->agf_refcount_root;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_refcountbt_key_diff(
|
xfs_refcountbt_key_diff(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key)
|
union xfs_btree_key *key)
|
||||||
@ -210,16 +210,16 @@ xfs_refcountbt_key_diff(
|
|||||||
struct xfs_refcount_irec *rec = &cur->bc_rec.rc;
|
struct xfs_refcount_irec *rec = &cur->bc_rec.rc;
|
||||||
struct xfs_refcount_key *kp = &key->refc;
|
struct xfs_refcount_key *kp = &key->refc;
|
||||||
|
|
||||||
return (__int64_t)be32_to_cpu(kp->rc_startblock) - rec->rc_startblock;
|
return (int64_t)be32_to_cpu(kp->rc_startblock) - rec->rc_startblock;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_refcountbt_diff_two_keys(
|
xfs_refcountbt_diff_two_keys(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *k1,
|
union xfs_btree_key *k1,
|
||||||
union xfs_btree_key *k2)
|
union xfs_btree_key *k2)
|
||||||
{
|
{
|
||||||
return (__int64_t)be32_to_cpu(k1->refc.rc_startblock) -
|
return (int64_t)be32_to_cpu(k1->refc.rc_startblock) -
|
||||||
be32_to_cpu(k2->refc.rc_startblock);
|
be32_to_cpu(k2->refc.rc_startblock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -285,7 +285,6 @@ const struct xfs_buf_ops xfs_refcountbt_buf_ops = {
|
|||||||
.verify_write = xfs_refcountbt_write_verify,
|
.verify_write = xfs_refcountbt_write_verify,
|
||||||
};
|
};
|
||||||
|
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
STATIC int
|
STATIC int
|
||||||
xfs_refcountbt_keys_inorder(
|
xfs_refcountbt_keys_inorder(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
@ -306,7 +305,6 @@ xfs_refcountbt_recs_inorder(
|
|||||||
be32_to_cpu(r1->refc.rc_blockcount) <=
|
be32_to_cpu(r1->refc.rc_blockcount) <=
|
||||||
be32_to_cpu(r2->refc.rc_startblock);
|
be32_to_cpu(r2->refc.rc_startblock);
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
static const struct xfs_btree_ops xfs_refcountbt_ops = {
|
static const struct xfs_btree_ops xfs_refcountbt_ops = {
|
||||||
.rec_len = sizeof(struct xfs_refcount_rec),
|
.rec_len = sizeof(struct xfs_refcount_rec),
|
||||||
@ -325,10 +323,8 @@ static const struct xfs_btree_ops xfs_refcountbt_ops = {
|
|||||||
.key_diff = xfs_refcountbt_key_diff,
|
.key_diff = xfs_refcountbt_key_diff,
|
||||||
.buf_ops = &xfs_refcountbt_buf_ops,
|
.buf_ops = &xfs_refcountbt_buf_ops,
|
||||||
.diff_two_keys = xfs_refcountbt_diff_two_keys,
|
.diff_two_keys = xfs_refcountbt_diff_two_keys,
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
.keys_inorder = xfs_refcountbt_keys_inorder,
|
.keys_inorder = xfs_refcountbt_keys_inorder,
|
||||||
.recs_inorder = xfs_refcountbt_recs_inorder,
|
.recs_inorder = xfs_refcountbt_recs_inorder,
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -179,7 +179,8 @@ done:
|
|||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
/* Convert an internal btree record to an rmap record. */
|
||||||
|
int
|
||||||
xfs_rmap_btrec_to_irec(
|
xfs_rmap_btrec_to_irec(
|
||||||
union xfs_btree_rec *rec,
|
union xfs_btree_rec *rec,
|
||||||
struct xfs_rmap_irec *irec)
|
struct xfs_rmap_irec *irec)
|
||||||
@ -2061,7 +2062,7 @@ int
|
|||||||
xfs_rmap_finish_one(
|
xfs_rmap_finish_one(
|
||||||
struct xfs_trans *tp,
|
struct xfs_trans *tp,
|
||||||
enum xfs_rmap_intent_type type,
|
enum xfs_rmap_intent_type type,
|
||||||
__uint64_t owner,
|
uint64_t owner,
|
||||||
int whichfork,
|
int whichfork,
|
||||||
xfs_fileoff_t startoff,
|
xfs_fileoff_t startoff,
|
||||||
xfs_fsblock_t startblock,
|
xfs_fsblock_t startblock,
|
||||||
@ -2086,8 +2087,7 @@ xfs_rmap_finish_one(
|
|||||||
startoff, blockcount, state);
|
startoff, blockcount, state);
|
||||||
|
|
||||||
if (XFS_TEST_ERROR(false, mp,
|
if (XFS_TEST_ERROR(false, mp,
|
||||||
XFS_ERRTAG_RMAP_FINISH_ONE,
|
XFS_ERRTAG_RMAP_FINISH_ONE))
|
||||||
XFS_RANDOM_RMAP_FINISH_ONE))
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2182,7 +2182,7 @@ __xfs_rmap_add(
|
|||||||
struct xfs_mount *mp,
|
struct xfs_mount *mp,
|
||||||
struct xfs_defer_ops *dfops,
|
struct xfs_defer_ops *dfops,
|
||||||
enum xfs_rmap_intent_type type,
|
enum xfs_rmap_intent_type type,
|
||||||
__uint64_t owner,
|
uint64_t owner,
|
||||||
int whichfork,
|
int whichfork,
|
||||||
struct xfs_bmbt_irec *bmap)
|
struct xfs_bmbt_irec *bmap)
|
||||||
{
|
{
|
||||||
@ -2266,7 +2266,7 @@ xfs_rmap_alloc_extent(
|
|||||||
xfs_agnumber_t agno,
|
xfs_agnumber_t agno,
|
||||||
xfs_agblock_t bno,
|
xfs_agblock_t bno,
|
||||||
xfs_extlen_t len,
|
xfs_extlen_t len,
|
||||||
__uint64_t owner)
|
uint64_t owner)
|
||||||
{
|
{
|
||||||
struct xfs_bmbt_irec bmap;
|
struct xfs_bmbt_irec bmap;
|
||||||
|
|
||||||
@ -2290,7 +2290,7 @@ xfs_rmap_free_extent(
|
|||||||
xfs_agnumber_t agno,
|
xfs_agnumber_t agno,
|
||||||
xfs_agblock_t bno,
|
xfs_agblock_t bno,
|
||||||
xfs_extlen_t len,
|
xfs_extlen_t len,
|
||||||
__uint64_t owner)
|
uint64_t owner)
|
||||||
{
|
{
|
||||||
struct xfs_bmbt_irec bmap;
|
struct xfs_bmbt_irec bmap;
|
||||||
|
|
||||||
|
@ -179,7 +179,7 @@ enum xfs_rmap_intent_type {
|
|||||||
struct xfs_rmap_intent {
|
struct xfs_rmap_intent {
|
||||||
struct list_head ri_list;
|
struct list_head ri_list;
|
||||||
enum xfs_rmap_intent_type ri_type;
|
enum xfs_rmap_intent_type ri_type;
|
||||||
__uint64_t ri_owner;
|
uint64_t ri_owner;
|
||||||
int ri_whichfork;
|
int ri_whichfork;
|
||||||
struct xfs_bmbt_irec ri_bmap;
|
struct xfs_bmbt_irec ri_bmap;
|
||||||
};
|
};
|
||||||
@ -196,15 +196,15 @@ int xfs_rmap_convert_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
|||||||
struct xfs_bmbt_irec *imap);
|
struct xfs_bmbt_irec *imap);
|
||||||
int xfs_rmap_alloc_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
int xfs_rmap_alloc_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
||||||
xfs_agnumber_t agno, xfs_agblock_t bno, xfs_extlen_t len,
|
xfs_agnumber_t agno, xfs_agblock_t bno, xfs_extlen_t len,
|
||||||
__uint64_t owner);
|
uint64_t owner);
|
||||||
int xfs_rmap_free_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
int xfs_rmap_free_extent(struct xfs_mount *mp, struct xfs_defer_ops *dfops,
|
||||||
xfs_agnumber_t agno, xfs_agblock_t bno, xfs_extlen_t len,
|
xfs_agnumber_t agno, xfs_agblock_t bno, xfs_extlen_t len,
|
||||||
__uint64_t owner);
|
uint64_t owner);
|
||||||
|
|
||||||
void xfs_rmap_finish_one_cleanup(struct xfs_trans *tp,
|
void xfs_rmap_finish_one_cleanup(struct xfs_trans *tp,
|
||||||
struct xfs_btree_cur *rcur, int error);
|
struct xfs_btree_cur *rcur, int error);
|
||||||
int xfs_rmap_finish_one(struct xfs_trans *tp, enum xfs_rmap_intent_type type,
|
int xfs_rmap_finish_one(struct xfs_trans *tp, enum xfs_rmap_intent_type type,
|
||||||
__uint64_t owner, int whichfork, xfs_fileoff_t startoff,
|
uint64_t owner, int whichfork, xfs_fileoff_t startoff,
|
||||||
xfs_fsblock_t startblock, xfs_filblks_t blockcount,
|
xfs_fsblock_t startblock, xfs_filblks_t blockcount,
|
||||||
xfs_exntst_t state, struct xfs_btree_cur **pcur);
|
xfs_exntst_t state, struct xfs_btree_cur **pcur);
|
||||||
|
|
||||||
@ -216,5 +216,8 @@ int xfs_rmap_lookup_le_range(struct xfs_btree_cur *cur, xfs_agblock_t bno,
|
|||||||
struct xfs_rmap_irec *irec, int *stat);
|
struct xfs_rmap_irec *irec, int *stat);
|
||||||
int xfs_rmap_compare(const struct xfs_rmap_irec *a,
|
int xfs_rmap_compare(const struct xfs_rmap_irec *a,
|
||||||
const struct xfs_rmap_irec *b);
|
const struct xfs_rmap_irec *b);
|
||||||
|
union xfs_btree_rec;
|
||||||
|
int xfs_rmap_btrec_to_irec(union xfs_btree_rec *rec,
|
||||||
|
struct xfs_rmap_irec *irec);
|
||||||
|
|
||||||
#endif /* __XFS_RMAP_H__ */
|
#endif /* __XFS_RMAP_H__ */
|
||||||
|
@ -199,7 +199,7 @@ xfs_rmapbt_init_high_key_from_rec(
|
|||||||
union xfs_btree_key *key,
|
union xfs_btree_key *key,
|
||||||
union xfs_btree_rec *rec)
|
union xfs_btree_rec *rec)
|
||||||
{
|
{
|
||||||
__uint64_t off;
|
uint64_t off;
|
||||||
int adj;
|
int adj;
|
||||||
|
|
||||||
adj = be32_to_cpu(rec->rmap.rm_blockcount) - 1;
|
adj = be32_to_cpu(rec->rmap.rm_blockcount) - 1;
|
||||||
@ -241,7 +241,7 @@ xfs_rmapbt_init_ptr_from_cur(
|
|||||||
ptr->s = agf->agf_roots[cur->bc_btnum];
|
ptr->s = agf->agf_roots[cur->bc_btnum];
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_rmapbt_key_diff(
|
xfs_rmapbt_key_diff(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *key)
|
union xfs_btree_key *key)
|
||||||
@ -249,9 +249,9 @@ xfs_rmapbt_key_diff(
|
|||||||
struct xfs_rmap_irec *rec = &cur->bc_rec.r;
|
struct xfs_rmap_irec *rec = &cur->bc_rec.r;
|
||||||
struct xfs_rmap_key *kp = &key->rmap;
|
struct xfs_rmap_key *kp = &key->rmap;
|
||||||
__u64 x, y;
|
__u64 x, y;
|
||||||
__int64_t d;
|
int64_t d;
|
||||||
|
|
||||||
d = (__int64_t)be32_to_cpu(kp->rm_startblock) - rec->rm_startblock;
|
d = (int64_t)be32_to_cpu(kp->rm_startblock) - rec->rm_startblock;
|
||||||
if (d)
|
if (d)
|
||||||
return d;
|
return d;
|
||||||
|
|
||||||
@ -271,7 +271,7 @@ xfs_rmapbt_key_diff(
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC __int64_t
|
STATIC int64_t
|
||||||
xfs_rmapbt_diff_two_keys(
|
xfs_rmapbt_diff_two_keys(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *k1,
|
union xfs_btree_key *k1,
|
||||||
@ -279,10 +279,10 @@ xfs_rmapbt_diff_two_keys(
|
|||||||
{
|
{
|
||||||
struct xfs_rmap_key *kp1 = &k1->rmap;
|
struct xfs_rmap_key *kp1 = &k1->rmap;
|
||||||
struct xfs_rmap_key *kp2 = &k2->rmap;
|
struct xfs_rmap_key *kp2 = &k2->rmap;
|
||||||
__int64_t d;
|
int64_t d;
|
||||||
__u64 x, y;
|
__u64 x, y;
|
||||||
|
|
||||||
d = (__int64_t)be32_to_cpu(kp1->rm_startblock) -
|
d = (int64_t)be32_to_cpu(kp1->rm_startblock) -
|
||||||
be32_to_cpu(kp2->rm_startblock);
|
be32_to_cpu(kp2->rm_startblock);
|
||||||
if (d)
|
if (d)
|
||||||
return d;
|
return d;
|
||||||
@ -377,17 +377,16 @@ const struct xfs_buf_ops xfs_rmapbt_buf_ops = {
|
|||||||
.verify_write = xfs_rmapbt_write_verify,
|
.verify_write = xfs_rmapbt_write_verify,
|
||||||
};
|
};
|
||||||
|
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
STATIC int
|
STATIC int
|
||||||
xfs_rmapbt_keys_inorder(
|
xfs_rmapbt_keys_inorder(
|
||||||
struct xfs_btree_cur *cur,
|
struct xfs_btree_cur *cur,
|
||||||
union xfs_btree_key *k1,
|
union xfs_btree_key *k1,
|
||||||
union xfs_btree_key *k2)
|
union xfs_btree_key *k2)
|
||||||
{
|
{
|
||||||
__uint32_t x;
|
uint32_t x;
|
||||||
__uint32_t y;
|
uint32_t y;
|
||||||
__uint64_t a;
|
uint64_t a;
|
||||||
__uint64_t b;
|
uint64_t b;
|
||||||
|
|
||||||
x = be32_to_cpu(k1->rmap.rm_startblock);
|
x = be32_to_cpu(k1->rmap.rm_startblock);
|
||||||
y = be32_to_cpu(k2->rmap.rm_startblock);
|
y = be32_to_cpu(k2->rmap.rm_startblock);
|
||||||
@ -414,10 +413,10 @@ xfs_rmapbt_recs_inorder(
|
|||||||
union xfs_btree_rec *r1,
|
union xfs_btree_rec *r1,
|
||||||
union xfs_btree_rec *r2)
|
union xfs_btree_rec *r2)
|
||||||
{
|
{
|
||||||
__uint32_t x;
|
uint32_t x;
|
||||||
__uint32_t y;
|
uint32_t y;
|
||||||
__uint64_t a;
|
uint64_t a;
|
||||||
__uint64_t b;
|
uint64_t b;
|
||||||
|
|
||||||
x = be32_to_cpu(r1->rmap.rm_startblock);
|
x = be32_to_cpu(r1->rmap.rm_startblock);
|
||||||
y = be32_to_cpu(r2->rmap.rm_startblock);
|
y = be32_to_cpu(r2->rmap.rm_startblock);
|
||||||
@ -437,7 +436,6 @@ xfs_rmapbt_recs_inorder(
|
|||||||
return 1;
|
return 1;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif /* DEBUG */
|
|
||||||
|
|
||||||
static const struct xfs_btree_ops xfs_rmapbt_ops = {
|
static const struct xfs_btree_ops xfs_rmapbt_ops = {
|
||||||
.rec_len = sizeof(struct xfs_rmap_rec),
|
.rec_len = sizeof(struct xfs_rmap_rec),
|
||||||
@ -456,10 +454,8 @@ static const struct xfs_btree_ops xfs_rmapbt_ops = {
|
|||||||
.key_diff = xfs_rmapbt_key_diff,
|
.key_diff = xfs_rmapbt_key_diff,
|
||||||
.buf_ops = &xfs_rmapbt_buf_ops,
|
.buf_ops = &xfs_rmapbt_buf_ops,
|
||||||
.diff_two_keys = xfs_rmapbt_diff_two_keys,
|
.diff_two_keys = xfs_rmapbt_diff_two_keys,
|
||||||
#if defined(DEBUG) || defined(XFS_WARN)
|
|
||||||
.keys_inorder = xfs_rmapbt_keys_inorder,
|
.keys_inorder = xfs_rmapbt_keys_inorder,
|
||||||
.recs_inorder = xfs_rmapbt_recs_inorder,
|
.recs_inorder = xfs_rmapbt_recs_inorder,
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -70,7 +70,7 @@ const struct xfs_buf_ops xfs_rtbuf_ops = {
|
|||||||
* Get a buffer for the bitmap or summary file block specified.
|
* Get a buffer for the bitmap or summary file block specified.
|
||||||
* The buffer is returned read and locked.
|
* The buffer is returned read and locked.
|
||||||
*/
|
*/
|
||||||
static int
|
int
|
||||||
xfs_rtbuf_get(
|
xfs_rtbuf_get(
|
||||||
xfs_mount_t *mp, /* file system mount structure */
|
xfs_mount_t *mp, /* file system mount structure */
|
||||||
xfs_trans_t *tp, /* transaction pointer */
|
xfs_trans_t *tp, /* transaction pointer */
|
||||||
@ -1011,7 +1011,7 @@ xfs_rtfree_extent(
|
|||||||
mp->m_sb.sb_rextents) {
|
mp->m_sb.sb_rextents) {
|
||||||
if (!(mp->m_rbmip->i_d.di_flags & XFS_DIFLAG_NEWRTBM))
|
if (!(mp->m_rbmip->i_d.di_flags & XFS_DIFLAG_NEWRTBM))
|
||||||
mp->m_rbmip->i_d.di_flags |= XFS_DIFLAG_NEWRTBM;
|
mp->m_rbmip->i_d.di_flags |= XFS_DIFLAG_NEWRTBM;
|
||||||
*(__uint64_t *)&VFS_I(mp->m_rbmip)->i_atime = 0;
|
*(uint64_t *)&VFS_I(mp->m_rbmip)->i_atime = 0;
|
||||||
xfs_trans_log_inode(tp, mp->m_rbmip, XFS_ILOG_CORE);
|
xfs_trans_log_inode(tp, mp->m_rbmip, XFS_ILOG_CORE);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -448,7 +448,7 @@ xfs_sb_quota_to_disk(
|
|||||||
struct xfs_dsb *to,
|
struct xfs_dsb *to,
|
||||||
struct xfs_sb *from)
|
struct xfs_sb *from)
|
||||||
{
|
{
|
||||||
__uint16_t qflags = from->sb_qflags;
|
uint16_t qflags = from->sb_qflags;
|
||||||
|
|
||||||
to->sb_uquotino = cpu_to_be64(from->sb_uquotino);
|
to->sb_uquotino = cpu_to_be64(from->sb_uquotino);
|
||||||
if (xfs_sb_version_has_pquotino(from)) {
|
if (xfs_sb_version_has_pquotino(from)) {
|
||||||
@ -756,7 +756,7 @@ xfs_sb_mount_common(
|
|||||||
mp->m_refc_mnr[1] = mp->m_refc_mxr[1] / 2;
|
mp->m_refc_mnr[1] = mp->m_refc_mxr[1] / 2;
|
||||||
|
|
||||||
mp->m_bsize = XFS_FSB_TO_BB(mp, 1);
|
mp->m_bsize = XFS_FSB_TO_BB(mp, 1);
|
||||||
mp->m_ialloc_inos = (int)MAX((__uint16_t)XFS_INODES_PER_CHUNK,
|
mp->m_ialloc_inos = (int)MAX((uint16_t)XFS_INODES_PER_CHUNK,
|
||||||
sbp->sb_inopblock);
|
sbp->sb_inopblock);
|
||||||
mp->m_ialloc_blks = mp->m_ialloc_inos >> sbp->sb_inopblog;
|
mp->m_ialloc_blks = mp->m_ialloc_inos >> sbp->sb_inopblog;
|
||||||
|
|
||||||
|
@ -114,7 +114,7 @@ xfs_symlink_verify(
|
|||||||
if (bp->b_bn != be64_to_cpu(dsl->sl_blkno))
|
if (bp->b_bn != be64_to_cpu(dsl->sl_blkno))
|
||||||
return false;
|
return false;
|
||||||
if (be32_to_cpu(dsl->sl_offset) +
|
if (be32_to_cpu(dsl->sl_offset) +
|
||||||
be32_to_cpu(dsl->sl_bytes) >= MAXPATHLEN)
|
be32_to_cpu(dsl->sl_bytes) >= XFS_SYMLINK_MAXLEN)
|
||||||
return false;
|
return false;
|
||||||
if (dsl->sl_owner == 0)
|
if (dsl->sl_owner == 0)
|
||||||
return false;
|
return false;
|
||||||
|
@ -477,14 +477,14 @@ xfs_calc_mkdir_reservation(
|
|||||||
/*
|
/*
|
||||||
* Making a new symplink is the same as creating a new file, but
|
* Making a new symplink is the same as creating a new file, but
|
||||||
* with the added blocks for remote symlink data which can be up to 1kB in
|
* with the added blocks for remote symlink data which can be up to 1kB in
|
||||||
* length (MAXPATHLEN).
|
* length (XFS_SYMLINK_MAXLEN).
|
||||||
*/
|
*/
|
||||||
STATIC uint
|
STATIC uint
|
||||||
xfs_calc_symlink_reservation(
|
xfs_calc_symlink_reservation(
|
||||||
struct xfs_mount *mp)
|
struct xfs_mount *mp)
|
||||||
{
|
{
|
||||||
return xfs_calc_create_reservation(mp) +
|
return xfs_calc_create_reservation(mp) +
|
||||||
xfs_calc_buf_res(1, MAXPATHLEN);
|
xfs_calc_buf_res(1, XFS_SYMLINK_MAXLEN);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -18,34 +18,34 @@
|
|||||||
#ifndef __XFS_TYPES_H__
|
#ifndef __XFS_TYPES_H__
|
||||||
#define __XFS_TYPES_H__
|
#define __XFS_TYPES_H__
|
||||||
|
|
||||||
typedef __uint32_t prid_t; /* project ID */
|
typedef uint32_t prid_t; /* project ID */
|
||||||
|
|
||||||
typedef __uint32_t xfs_agblock_t; /* blockno in alloc. group */
|
typedef uint32_t xfs_agblock_t; /* blockno in alloc. group */
|
||||||
typedef __uint32_t xfs_agino_t; /* inode # within allocation grp */
|
typedef uint32_t xfs_agino_t; /* inode # within allocation grp */
|
||||||
typedef __uint32_t xfs_extlen_t; /* extent length in blocks */
|
typedef uint32_t xfs_extlen_t; /* extent length in blocks */
|
||||||
typedef __uint32_t xfs_agnumber_t; /* allocation group number */
|
typedef uint32_t xfs_agnumber_t; /* allocation group number */
|
||||||
typedef __int32_t xfs_extnum_t; /* # of extents in a file */
|
typedef int32_t xfs_extnum_t; /* # of extents in a file */
|
||||||
typedef __int16_t xfs_aextnum_t; /* # extents in an attribute fork */
|
typedef int16_t xfs_aextnum_t; /* # extents in an attribute fork */
|
||||||
typedef __int64_t xfs_fsize_t; /* bytes in a file */
|
typedef int64_t xfs_fsize_t; /* bytes in a file */
|
||||||
typedef __uint64_t xfs_ufsize_t; /* unsigned bytes in a file */
|
typedef uint64_t xfs_ufsize_t; /* unsigned bytes in a file */
|
||||||
|
|
||||||
typedef __int32_t xfs_suminfo_t; /* type of bitmap summary info */
|
typedef int32_t xfs_suminfo_t; /* type of bitmap summary info */
|
||||||
typedef __int32_t xfs_rtword_t; /* word type for bitmap manipulations */
|
typedef int32_t xfs_rtword_t; /* word type for bitmap manipulations */
|
||||||
|
|
||||||
typedef __int64_t xfs_lsn_t; /* log sequence number */
|
typedef int64_t xfs_lsn_t; /* log sequence number */
|
||||||
typedef __int32_t xfs_tid_t; /* transaction identifier */
|
typedef int32_t xfs_tid_t; /* transaction identifier */
|
||||||
|
|
||||||
typedef __uint32_t xfs_dablk_t; /* dir/attr block number (in file) */
|
typedef uint32_t xfs_dablk_t; /* dir/attr block number (in file) */
|
||||||
typedef __uint32_t xfs_dahash_t; /* dir/attr hash value */
|
typedef uint32_t xfs_dahash_t; /* dir/attr hash value */
|
||||||
|
|
||||||
typedef __uint64_t xfs_fsblock_t; /* blockno in filesystem (agno|agbno) */
|
typedef uint64_t xfs_fsblock_t; /* blockno in filesystem (agno|agbno) */
|
||||||
typedef __uint64_t xfs_rfsblock_t; /* blockno in filesystem (raw) */
|
typedef uint64_t xfs_rfsblock_t; /* blockno in filesystem (raw) */
|
||||||
typedef __uint64_t xfs_rtblock_t; /* extent (block) in realtime area */
|
typedef uint64_t xfs_rtblock_t; /* extent (block) in realtime area */
|
||||||
typedef __uint64_t xfs_fileoff_t; /* block number in a file */
|
typedef uint64_t xfs_fileoff_t; /* block number in a file */
|
||||||
typedef __uint64_t xfs_filblks_t; /* number of blocks in a file */
|
typedef uint64_t xfs_filblks_t; /* number of blocks in a file */
|
||||||
|
|
||||||
typedef __int64_t xfs_srtblock_t; /* signed version of xfs_rtblock_t */
|
typedef int64_t xfs_srtblock_t; /* signed version of xfs_rtblock_t */
|
||||||
typedef __int64_t xfs_sfiloff_t; /* signed block number in a file */
|
typedef int64_t xfs_sfiloff_t; /* signed block number in a file */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Null values for the types.
|
* Null values for the types.
|
||||||
@ -125,7 +125,7 @@ struct xfs_name {
|
|||||||
* uid_t and gid_t are hard-coded to 32 bits in the inode.
|
* uid_t and gid_t are hard-coded to 32 bits in the inode.
|
||||||
* Hence, an 'id' in a dquot is 32 bits..
|
* Hence, an 'id' in a dquot is 32 bits..
|
||||||
*/
|
*/
|
||||||
typedef __uint32_t xfs_dqid_t;
|
typedef uint32_t xfs_dqid_t;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Constants for bit manipulations.
|
* Constants for bit manipulations.
|
||||||
|
@ -24,6 +24,10 @@
|
|||||||
#define XFS_BUF_LOCK_TRACKING 1
|
#define XFS_BUF_LOCK_TRACKING 1
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_XFS_ASSERT_FATAL
|
||||||
|
#define XFS_ASSERT_FATAL 1
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_XFS_WARN
|
#ifdef CONFIG_XFS_WARN
|
||||||
#define XFS_WARN 1
|
#define XFS_WARN 1
|
||||||
#endif
|
#endif
|
||||||
|
@ -170,8 +170,8 @@ xfs_get_acl(struct inode *inode, int type)
|
|||||||
return acl;
|
return acl;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC int
|
int
|
||||||
__xfs_set_acl(struct inode *inode, int type, struct posix_acl *acl)
|
__xfs_set_acl(struct inode *inode, struct posix_acl *acl, int type)
|
||||||
{
|
{
|
||||||
struct xfs_inode *ip = XFS_I(inode);
|
struct xfs_inode *ip = XFS_I(inode);
|
||||||
unsigned char *ea_name;
|
unsigned char *ea_name;
|
||||||
@ -268,5 +268,5 @@ xfs_set_acl(struct inode *inode, struct posix_acl *acl, int type)
|
|||||||
}
|
}
|
||||||
|
|
||||||
set_acl:
|
set_acl:
|
||||||
return __xfs_set_acl(inode, type, acl);
|
return __xfs_set_acl(inode, acl, type);
|
||||||
}
|
}
|
||||||
|
@ -24,6 +24,7 @@ struct posix_acl;
|
|||||||
#ifdef CONFIG_XFS_POSIX_ACL
|
#ifdef CONFIG_XFS_POSIX_ACL
|
||||||
extern struct posix_acl *xfs_get_acl(struct inode *inode, int type);
|
extern struct posix_acl *xfs_get_acl(struct inode *inode, int type);
|
||||||
extern int xfs_set_acl(struct inode *inode, struct posix_acl *acl, int type);
|
extern int xfs_set_acl(struct inode *inode, struct posix_acl *acl, int type);
|
||||||
|
extern int __xfs_set_acl(struct inode *inode, struct posix_acl *acl, int type);
|
||||||
#else
|
#else
|
||||||
static inline struct posix_acl *xfs_get_acl(struct inode *inode, int type)
|
static inline struct posix_acl *xfs_get_acl(struct inode *inode, int type)
|
||||||
{
|
{
|
||||||
|
@ -839,7 +839,7 @@ xfs_writepage_map(
|
|||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
struct page *page,
|
struct page *page,
|
||||||
loff_t offset,
|
loff_t offset,
|
||||||
__uint64_t end_offset)
|
uint64_t end_offset)
|
||||||
{
|
{
|
||||||
LIST_HEAD(submit_list);
|
LIST_HEAD(submit_list);
|
||||||
struct xfs_ioend *ioend, *next;
|
struct xfs_ioend *ioend, *next;
|
||||||
@ -994,7 +994,7 @@ xfs_do_writepage(
|
|||||||
struct xfs_writepage_ctx *wpc = data;
|
struct xfs_writepage_ctx *wpc = data;
|
||||||
struct inode *inode = page->mapping->host;
|
struct inode *inode = page->mapping->host;
|
||||||
loff_t offset;
|
loff_t offset;
|
||||||
__uint64_t end_offset;
|
uint64_t end_offset;
|
||||||
pgoff_t end_index;
|
pgoff_t end_index;
|
||||||
|
|
||||||
trace_xfs_writepage(inode, page, 0, 0);
|
trace_xfs_writepage(inode, page, 0, 0);
|
||||||
|
@ -117,6 +117,7 @@ typedef void (*put_listent_func_t)(struct xfs_attr_list_context *, int,
|
|||||||
unsigned char *, int, int);
|
unsigned char *, int, int);
|
||||||
|
|
||||||
typedef struct xfs_attr_list_context {
|
typedef struct xfs_attr_list_context {
|
||||||
|
struct xfs_trans *tp;
|
||||||
struct xfs_inode *dp; /* inode */
|
struct xfs_inode *dp; /* inode */
|
||||||
struct attrlist_cursor_kern *cursor; /* position in list */
|
struct attrlist_cursor_kern *cursor; /* position in list */
|
||||||
char *alist; /* output buffer */
|
char *alist; /* output buffer */
|
||||||
@ -140,8 +141,10 @@ typedef struct xfs_attr_list_context {
|
|||||||
* Overall external interface routines.
|
* Overall external interface routines.
|
||||||
*/
|
*/
|
||||||
int xfs_attr_inactive(struct xfs_inode *dp);
|
int xfs_attr_inactive(struct xfs_inode *dp);
|
||||||
|
int xfs_attr_list_int_ilocked(struct xfs_attr_list_context *);
|
||||||
int xfs_attr_list_int(struct xfs_attr_list_context *);
|
int xfs_attr_list_int(struct xfs_attr_list_context *);
|
||||||
int xfs_inode_hasattr(struct xfs_inode *ip);
|
int xfs_inode_hasattr(struct xfs_inode *ip);
|
||||||
|
int xfs_attr_get_ilocked(struct xfs_inode *ip, struct xfs_da_args *args);
|
||||||
int xfs_attr_get(struct xfs_inode *ip, const unsigned char *name,
|
int xfs_attr_get(struct xfs_inode *ip, const unsigned char *name,
|
||||||
unsigned char *value, int *valuelenp, int flags);
|
unsigned char *value, int *valuelenp, int flags);
|
||||||
int xfs_attr_set(struct xfs_inode *dp, const unsigned char *name,
|
int xfs_attr_set(struct xfs_inode *dp, const unsigned char *name,
|
||||||
|
@ -230,7 +230,7 @@ xfs_attr_node_list(xfs_attr_list_context_t *context)
|
|||||||
*/
|
*/
|
||||||
bp = NULL;
|
bp = NULL;
|
||||||
if (cursor->blkno > 0) {
|
if (cursor->blkno > 0) {
|
||||||
error = xfs_da3_node_read(NULL, dp, cursor->blkno, -1,
|
error = xfs_da3_node_read(context->tp, dp, cursor->blkno, -1,
|
||||||
&bp, XFS_ATTR_FORK);
|
&bp, XFS_ATTR_FORK);
|
||||||
if ((error != 0) && (error != -EFSCORRUPTED))
|
if ((error != 0) && (error != -EFSCORRUPTED))
|
||||||
return error;
|
return error;
|
||||||
@ -242,7 +242,7 @@ xfs_attr_node_list(xfs_attr_list_context_t *context)
|
|||||||
case XFS_DA_NODE_MAGIC:
|
case XFS_DA_NODE_MAGIC:
|
||||||
case XFS_DA3_NODE_MAGIC:
|
case XFS_DA3_NODE_MAGIC:
|
||||||
trace_xfs_attr_list_wrong_blk(context);
|
trace_xfs_attr_list_wrong_blk(context);
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
bp = NULL;
|
bp = NULL;
|
||||||
break;
|
break;
|
||||||
case XFS_ATTR_LEAF_MAGIC:
|
case XFS_ATTR_LEAF_MAGIC:
|
||||||
@ -254,18 +254,18 @@ xfs_attr_node_list(xfs_attr_list_context_t *context)
|
|||||||
if (cursor->hashval > be32_to_cpu(
|
if (cursor->hashval > be32_to_cpu(
|
||||||
entries[leafhdr.count - 1].hashval)) {
|
entries[leafhdr.count - 1].hashval)) {
|
||||||
trace_xfs_attr_list_wrong_blk(context);
|
trace_xfs_attr_list_wrong_blk(context);
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
bp = NULL;
|
bp = NULL;
|
||||||
} else if (cursor->hashval <= be32_to_cpu(
|
} else if (cursor->hashval <= be32_to_cpu(
|
||||||
entries[0].hashval)) {
|
entries[0].hashval)) {
|
||||||
trace_xfs_attr_list_wrong_blk(context);
|
trace_xfs_attr_list_wrong_blk(context);
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
bp = NULL;
|
bp = NULL;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
trace_xfs_attr_list_wrong_blk(context);
|
trace_xfs_attr_list_wrong_blk(context);
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
bp = NULL;
|
bp = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -279,9 +279,9 @@ xfs_attr_node_list(xfs_attr_list_context_t *context)
|
|||||||
if (bp == NULL) {
|
if (bp == NULL) {
|
||||||
cursor->blkno = 0;
|
cursor->blkno = 0;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
__uint16_t magic;
|
uint16_t magic;
|
||||||
|
|
||||||
error = xfs_da3_node_read(NULL, dp,
|
error = xfs_da3_node_read(context->tp, dp,
|
||||||
cursor->blkno, -1, &bp,
|
cursor->blkno, -1, &bp,
|
||||||
XFS_ATTR_FORK);
|
XFS_ATTR_FORK);
|
||||||
if (error)
|
if (error)
|
||||||
@ -297,7 +297,7 @@ xfs_attr_node_list(xfs_attr_list_context_t *context)
|
|||||||
XFS_ERRLEVEL_LOW,
|
XFS_ERRLEVEL_LOW,
|
||||||
context->dp->i_mount,
|
context->dp->i_mount,
|
||||||
node);
|
node);
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
return -EFSCORRUPTED;
|
return -EFSCORRUPTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -313,10 +313,10 @@ xfs_attr_node_list(xfs_attr_list_context_t *context)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (i == nodehdr.count) {
|
if (i == nodehdr.count) {
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ASSERT(bp != NULL);
|
ASSERT(bp != NULL);
|
||||||
@ -333,12 +333,12 @@ xfs_attr_node_list(xfs_attr_list_context_t *context)
|
|||||||
if (context->seen_enough || leafhdr.forw == 0)
|
if (context->seen_enough || leafhdr.forw == 0)
|
||||||
break;
|
break;
|
||||||
cursor->blkno = leafhdr.forw;
|
cursor->blkno = leafhdr.forw;
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
error = xfs_attr3_leaf_read(NULL, dp, cursor->blkno, -1, &bp);
|
error = xfs_attr3_leaf_read(context->tp, dp, cursor->blkno, -1, &bp);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -448,15 +448,33 @@ xfs_attr_leaf_list(xfs_attr_list_context_t *context)
|
|||||||
trace_xfs_attr_leaf_list(context);
|
trace_xfs_attr_leaf_list(context);
|
||||||
|
|
||||||
context->cursor->blkno = 0;
|
context->cursor->blkno = 0;
|
||||||
error = xfs_attr3_leaf_read(NULL, context->dp, 0, -1, &bp);
|
error = xfs_attr3_leaf_read(context->tp, context->dp, 0, -1, &bp);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
|
|
||||||
xfs_attr3_leaf_list_int(bp, context);
|
xfs_attr3_leaf_list_int(bp, context);
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(context->tp, bp);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xfs_attr_list_int_ilocked(
|
||||||
|
struct xfs_attr_list_context *context)
|
||||||
|
{
|
||||||
|
struct xfs_inode *dp = context->dp;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Decide on what work routines to call based on the inode size.
|
||||||
|
*/
|
||||||
|
if (!xfs_inode_hasattr(dp))
|
||||||
|
return 0;
|
||||||
|
else if (dp->i_d.di_aformat == XFS_DINODE_FMT_LOCAL)
|
||||||
|
return xfs_attr_shortform_list(context);
|
||||||
|
else if (xfs_bmap_one_block(dp, XFS_ATTR_FORK))
|
||||||
|
return xfs_attr_leaf_list(context);
|
||||||
|
return xfs_attr_node_list(context);
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
xfs_attr_list_int(
|
xfs_attr_list_int(
|
||||||
xfs_attr_list_context_t *context)
|
xfs_attr_list_context_t *context)
|
||||||
@ -470,19 +488,8 @@ xfs_attr_list_int(
|
|||||||
if (XFS_FORCED_SHUTDOWN(dp->i_mount))
|
if (XFS_FORCED_SHUTDOWN(dp->i_mount))
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
/*
|
|
||||||
* Decide on what work routines to call based on the inode size.
|
|
||||||
*/
|
|
||||||
lock_mode = xfs_ilock_attr_map_shared(dp);
|
lock_mode = xfs_ilock_attr_map_shared(dp);
|
||||||
if (!xfs_inode_hasattr(dp)) {
|
error = xfs_attr_list_int_ilocked(context);
|
||||||
error = 0;
|
|
||||||
} else if (dp->i_d.di_aformat == XFS_DINODE_FMT_LOCAL) {
|
|
||||||
error = xfs_attr_shortform_list(context);
|
|
||||||
} else if (xfs_bmap_one_block(dp, XFS_ATTR_FORK)) {
|
|
||||||
error = xfs_attr_leaf_list(context);
|
|
||||||
} else {
|
|
||||||
error = xfs_attr_node_list(context);
|
|
||||||
}
|
|
||||||
xfs_iunlock(dp, lock_mode);
|
xfs_iunlock(dp, lock_mode);
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
@ -396,6 +396,7 @@ xfs_bui_recover(
|
|||||||
struct xfs_map_extent *bmap;
|
struct xfs_map_extent *bmap;
|
||||||
xfs_fsblock_t startblock_fsb;
|
xfs_fsblock_t startblock_fsb;
|
||||||
xfs_fsblock_t inode_fsb;
|
xfs_fsblock_t inode_fsb;
|
||||||
|
xfs_filblks_t count;
|
||||||
bool op_ok;
|
bool op_ok;
|
||||||
struct xfs_bud_log_item *budp;
|
struct xfs_bud_log_item *budp;
|
||||||
enum xfs_bmap_intent_type type;
|
enum xfs_bmap_intent_type type;
|
||||||
@ -404,6 +405,7 @@ xfs_bui_recover(
|
|||||||
struct xfs_trans *tp;
|
struct xfs_trans *tp;
|
||||||
struct xfs_inode *ip = NULL;
|
struct xfs_inode *ip = NULL;
|
||||||
struct xfs_defer_ops dfops;
|
struct xfs_defer_ops dfops;
|
||||||
|
struct xfs_bmbt_irec irec;
|
||||||
xfs_fsblock_t firstfsb;
|
xfs_fsblock_t firstfsb;
|
||||||
|
|
||||||
ASSERT(!test_bit(XFS_BUI_RECOVERED, &buip->bui_flags));
|
ASSERT(!test_bit(XFS_BUI_RECOVERED, &buip->bui_flags));
|
||||||
@ -481,13 +483,24 @@ xfs_bui_recover(
|
|||||||
}
|
}
|
||||||
xfs_trans_ijoin(tp, ip, 0);
|
xfs_trans_ijoin(tp, ip, 0);
|
||||||
|
|
||||||
|
count = bmap->me_len;
|
||||||
error = xfs_trans_log_finish_bmap_update(tp, budp, &dfops, type,
|
error = xfs_trans_log_finish_bmap_update(tp, budp, &dfops, type,
|
||||||
ip, whichfork, bmap->me_startoff,
|
ip, whichfork, bmap->me_startoff,
|
||||||
bmap->me_startblock, bmap->me_len,
|
bmap->me_startblock, &count, state);
|
||||||
state);
|
|
||||||
if (error)
|
if (error)
|
||||||
goto err_dfops;
|
goto err_dfops;
|
||||||
|
|
||||||
|
if (count > 0) {
|
||||||
|
ASSERT(type == XFS_BMAP_UNMAP);
|
||||||
|
irec.br_startblock = bmap->me_startblock;
|
||||||
|
irec.br_blockcount = count;
|
||||||
|
irec.br_startoff = bmap->me_startoff;
|
||||||
|
irec.br_state = state;
|
||||||
|
error = xfs_bmap_unmap_extent(tp->t_mountp, &dfops, ip, &irec);
|
||||||
|
if (error)
|
||||||
|
goto err_dfops;
|
||||||
|
}
|
||||||
|
|
||||||
/* Finish transaction, free inodes. */
|
/* Finish transaction, free inodes. */
|
||||||
error = xfs_defer_finish(&tp, &dfops, NULL);
|
error = xfs_defer_finish(&tp, &dfops, NULL);
|
||||||
if (error)
|
if (error)
|
||||||
|
@ -219,20 +219,24 @@ xfs_bmap_eof(
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Count leaf blocks given a range of extent records.
|
* Count leaf blocks given a range of extent records. Delayed allocation
|
||||||
|
* extents are not counted towards the totals.
|
||||||
*/
|
*/
|
||||||
STATIC void
|
STATIC void
|
||||||
xfs_bmap_count_leaves(
|
xfs_bmap_count_leaves(
|
||||||
xfs_ifork_t *ifp,
|
struct xfs_ifork *ifp,
|
||||||
xfs_extnum_t idx,
|
xfs_extnum_t *numrecs,
|
||||||
int numrecs,
|
xfs_filblks_t *count)
|
||||||
int *count)
|
|
||||||
{
|
{
|
||||||
int b;
|
xfs_extnum_t i;
|
||||||
|
xfs_extnum_t nr_exts = xfs_iext_count(ifp);
|
||||||
|
|
||||||
for (b = 0; b < numrecs; b++) {
|
for (i = 0; i < nr_exts; i++) {
|
||||||
xfs_bmbt_rec_host_t *frp = xfs_iext_get_ext(ifp, idx + b);
|
xfs_bmbt_rec_host_t *frp = xfs_iext_get_ext(ifp, i);
|
||||||
*count += xfs_bmbt_get_blockcount(frp);
|
if (!isnullstartblock(xfs_bmbt_get_startblock(frp))) {
|
||||||
|
(*numrecs)++;
|
||||||
|
*count += xfs_bmbt_get_blockcount(frp);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -245,7 +249,7 @@ xfs_bmap_disk_count_leaves(
|
|||||||
struct xfs_mount *mp,
|
struct xfs_mount *mp,
|
||||||
struct xfs_btree_block *block,
|
struct xfs_btree_block *block,
|
||||||
int numrecs,
|
int numrecs,
|
||||||
int *count)
|
xfs_filblks_t *count)
|
||||||
{
|
{
|
||||||
int b;
|
int b;
|
||||||
xfs_bmbt_rec_t *frp;
|
xfs_bmbt_rec_t *frp;
|
||||||
@ -260,17 +264,18 @@ xfs_bmap_disk_count_leaves(
|
|||||||
* Recursively walks each level of a btree
|
* Recursively walks each level of a btree
|
||||||
* to count total fsblocks in use.
|
* to count total fsblocks in use.
|
||||||
*/
|
*/
|
||||||
STATIC int /* error */
|
STATIC int
|
||||||
xfs_bmap_count_tree(
|
xfs_bmap_count_tree(
|
||||||
xfs_mount_t *mp, /* file system mount point */
|
struct xfs_mount *mp,
|
||||||
xfs_trans_t *tp, /* transaction pointer */
|
struct xfs_trans *tp,
|
||||||
xfs_ifork_t *ifp, /* inode fork pointer */
|
struct xfs_ifork *ifp,
|
||||||
xfs_fsblock_t blockno, /* file system block number */
|
xfs_fsblock_t blockno,
|
||||||
int levelin, /* level in btree */
|
int levelin,
|
||||||
int *count) /* Count of blocks */
|
xfs_extnum_t *nextents,
|
||||||
|
xfs_filblks_t *count)
|
||||||
{
|
{
|
||||||
int error;
|
int error;
|
||||||
xfs_buf_t *bp, *nbp;
|
struct xfs_buf *bp, *nbp;
|
||||||
int level = levelin;
|
int level = levelin;
|
||||||
__be64 *pp;
|
__be64 *pp;
|
||||||
xfs_fsblock_t bno = blockno;
|
xfs_fsblock_t bno = blockno;
|
||||||
@ -303,8 +308,9 @@ xfs_bmap_count_tree(
|
|||||||
/* Dive to the next level */
|
/* Dive to the next level */
|
||||||
pp = XFS_BMBT_PTR_ADDR(mp, block, 1, mp->m_bmap_dmxr[1]);
|
pp = XFS_BMBT_PTR_ADDR(mp, block, 1, mp->m_bmap_dmxr[1]);
|
||||||
bno = be64_to_cpu(*pp);
|
bno = be64_to_cpu(*pp);
|
||||||
if (unlikely((error =
|
error = xfs_bmap_count_tree(mp, tp, ifp, bno, level, nextents,
|
||||||
xfs_bmap_count_tree(mp, tp, ifp, bno, level, count)) < 0)) {
|
count);
|
||||||
|
if (error) {
|
||||||
xfs_trans_brelse(tp, bp);
|
xfs_trans_brelse(tp, bp);
|
||||||
XFS_ERROR_REPORT("xfs_bmap_count_tree(1)",
|
XFS_ERROR_REPORT("xfs_bmap_count_tree(1)",
|
||||||
XFS_ERRLEVEL_LOW, mp);
|
XFS_ERRLEVEL_LOW, mp);
|
||||||
@ -316,6 +322,7 @@ xfs_bmap_count_tree(
|
|||||||
for (;;) {
|
for (;;) {
|
||||||
nextbno = be64_to_cpu(block->bb_u.l.bb_rightsib);
|
nextbno = be64_to_cpu(block->bb_u.l.bb_rightsib);
|
||||||
numrecs = be16_to_cpu(block->bb_numrecs);
|
numrecs = be16_to_cpu(block->bb_numrecs);
|
||||||
|
(*nextents) += numrecs;
|
||||||
xfs_bmap_disk_count_leaves(mp, block, numrecs, count);
|
xfs_bmap_disk_count_leaves(mp, block, numrecs, count);
|
||||||
xfs_trans_brelse(tp, bp);
|
xfs_trans_brelse(tp, bp);
|
||||||
if (nextbno == NULLFSBLOCK)
|
if (nextbno == NULLFSBLOCK)
|
||||||
@ -334,46 +341,64 @@ xfs_bmap_count_tree(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Count fsblocks of the given fork.
|
* Count fsblocks of the given fork. Delayed allocation extents are
|
||||||
|
* not counted towards the totals.
|
||||||
*/
|
*/
|
||||||
static int /* error */
|
int
|
||||||
xfs_bmap_count_blocks(
|
xfs_bmap_count_blocks(
|
||||||
xfs_trans_t *tp, /* transaction pointer */
|
struct xfs_trans *tp,
|
||||||
xfs_inode_t *ip, /* incore inode */
|
struct xfs_inode *ip,
|
||||||
int whichfork, /* data or attr fork */
|
int whichfork,
|
||||||
int *count) /* out: count of blocks */
|
xfs_extnum_t *nextents,
|
||||||
|
xfs_filblks_t *count)
|
||||||
{
|
{
|
||||||
struct xfs_btree_block *block; /* current btree block */
|
struct xfs_mount *mp; /* file system mount structure */
|
||||||
xfs_fsblock_t bno; /* block # of "block" */
|
|
||||||
xfs_ifork_t *ifp; /* fork structure */
|
|
||||||
int level; /* btree level, for checking */
|
|
||||||
xfs_mount_t *mp; /* file system mount structure */
|
|
||||||
__be64 *pp; /* pointer to block address */
|
__be64 *pp; /* pointer to block address */
|
||||||
|
struct xfs_btree_block *block; /* current btree block */
|
||||||
|
struct xfs_ifork *ifp; /* fork structure */
|
||||||
|
xfs_fsblock_t bno; /* block # of "block" */
|
||||||
|
int level; /* btree level, for checking */
|
||||||
|
int error;
|
||||||
|
|
||||||
bno = NULLFSBLOCK;
|
bno = NULLFSBLOCK;
|
||||||
mp = ip->i_mount;
|
mp = ip->i_mount;
|
||||||
|
*nextents = 0;
|
||||||
|
*count = 0;
|
||||||
ifp = XFS_IFORK_PTR(ip, whichfork);
|
ifp = XFS_IFORK_PTR(ip, whichfork);
|
||||||
if ( XFS_IFORK_FORMAT(ip, whichfork) == XFS_DINODE_FMT_EXTENTS ) {
|
if (!ifp)
|
||||||
xfs_bmap_count_leaves(ifp, 0, xfs_iext_count(ifp), count);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
switch (XFS_IFORK_FORMAT(ip, whichfork)) {
|
||||||
* Root level must use BMAP_BROOT_PTR_ADDR macro to get ptr out.
|
case XFS_DINODE_FMT_EXTENTS:
|
||||||
*/
|
xfs_bmap_count_leaves(ifp, nextents, count);
|
||||||
block = ifp->if_broot;
|
return 0;
|
||||||
level = be16_to_cpu(block->bb_level);
|
case XFS_DINODE_FMT_BTREE:
|
||||||
ASSERT(level > 0);
|
if (!(ifp->if_flags & XFS_IFEXTENTS)) {
|
||||||
pp = XFS_BMAP_BROOT_PTR_ADDR(mp, block, 1, ifp->if_broot_bytes);
|
error = xfs_iread_extents(tp, ip, whichfork);
|
||||||
bno = be64_to_cpu(*pp);
|
if (error)
|
||||||
ASSERT(bno != NULLFSBLOCK);
|
return error;
|
||||||
ASSERT(XFS_FSB_TO_AGNO(mp, bno) < mp->m_sb.sb_agcount);
|
}
|
||||||
ASSERT(XFS_FSB_TO_AGBNO(mp, bno) < mp->m_sb.sb_agblocks);
|
|
||||||
|
|
||||||
if (unlikely(xfs_bmap_count_tree(mp, tp, ifp, bno, level, count) < 0)) {
|
/*
|
||||||
XFS_ERROR_REPORT("xfs_bmap_count_blocks(2)", XFS_ERRLEVEL_LOW,
|
* Root level must use BMAP_BROOT_PTR_ADDR macro to get ptr out.
|
||||||
mp);
|
*/
|
||||||
return -EFSCORRUPTED;
|
block = ifp->if_broot;
|
||||||
|
level = be16_to_cpu(block->bb_level);
|
||||||
|
ASSERT(level > 0);
|
||||||
|
pp = XFS_BMAP_BROOT_PTR_ADDR(mp, block, 1, ifp->if_broot_bytes);
|
||||||
|
bno = be64_to_cpu(*pp);
|
||||||
|
ASSERT(bno != NULLFSBLOCK);
|
||||||
|
ASSERT(XFS_FSB_TO_AGNO(mp, bno) < mp->m_sb.sb_agcount);
|
||||||
|
ASSERT(XFS_FSB_TO_AGBNO(mp, bno) < mp->m_sb.sb_agblocks);
|
||||||
|
|
||||||
|
error = xfs_bmap_count_tree(mp, tp, ifp, bno, level,
|
||||||
|
nextents, count);
|
||||||
|
if (error) {
|
||||||
|
XFS_ERROR_REPORT("xfs_bmap_count_blocks(2)",
|
||||||
|
XFS_ERRLEVEL_LOW, mp);
|
||||||
|
return -EFSCORRUPTED;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -389,11 +414,11 @@ xfs_getbmapx_fix_eof_hole(
|
|||||||
struct getbmapx *out, /* output structure */
|
struct getbmapx *out, /* output structure */
|
||||||
int prealloced, /* this is a file with
|
int prealloced, /* this is a file with
|
||||||
* preallocated data space */
|
* preallocated data space */
|
||||||
__int64_t end, /* last block requested */
|
int64_t end, /* last block requested */
|
||||||
xfs_fsblock_t startblock,
|
xfs_fsblock_t startblock,
|
||||||
bool moretocome)
|
bool moretocome)
|
||||||
{
|
{
|
||||||
__int64_t fixlen;
|
int64_t fixlen;
|
||||||
xfs_mount_t *mp; /* file system mount point */
|
xfs_mount_t *mp; /* file system mount point */
|
||||||
xfs_ifork_t *ifp; /* inode fork pointer */
|
xfs_ifork_t *ifp; /* inode fork pointer */
|
||||||
xfs_extnum_t lastx; /* last extent pointer */
|
xfs_extnum_t lastx; /* last extent pointer */
|
||||||
@ -455,8 +480,8 @@ xfs_getbmap_adjust_shared(
|
|||||||
|
|
||||||
agno = XFS_FSB_TO_AGNO(mp, map->br_startblock);
|
agno = XFS_FSB_TO_AGNO(mp, map->br_startblock);
|
||||||
agbno = XFS_FSB_TO_AGBNO(mp, map->br_startblock);
|
agbno = XFS_FSB_TO_AGBNO(mp, map->br_startblock);
|
||||||
error = xfs_reflink_find_shared(mp, agno, agbno, map->br_blockcount,
|
error = xfs_reflink_find_shared(mp, NULL, agno, agbno,
|
||||||
&ebno, &elen, true);
|
map->br_blockcount, &ebno, &elen, true);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
|
|
||||||
@ -514,9 +539,9 @@ xfs_getbmap(
|
|||||||
xfs_bmap_format_t formatter, /* format to user */
|
xfs_bmap_format_t formatter, /* format to user */
|
||||||
void *arg) /* formatter arg */
|
void *arg) /* formatter arg */
|
||||||
{
|
{
|
||||||
__int64_t bmvend; /* last block requested */
|
int64_t bmvend; /* last block requested */
|
||||||
int error = 0; /* return value */
|
int error = 0; /* return value */
|
||||||
__int64_t fixlen; /* length for -1 case */
|
int64_t fixlen; /* length for -1 case */
|
||||||
int i; /* extent number */
|
int i; /* extent number */
|
||||||
int lock; /* lock state */
|
int lock; /* lock state */
|
||||||
xfs_bmbt_irec_t *map; /* buffer for user's data */
|
xfs_bmbt_irec_t *map; /* buffer for user's data */
|
||||||
@ -605,7 +630,7 @@ xfs_getbmap(
|
|||||||
if (bmv->bmv_length == -1) {
|
if (bmv->bmv_length == -1) {
|
||||||
fixlen = XFS_FSB_TO_BB(mp, XFS_B_TO_FSB(mp, fixlen));
|
fixlen = XFS_FSB_TO_BB(mp, XFS_B_TO_FSB(mp, fixlen));
|
||||||
bmv->bmv_length =
|
bmv->bmv_length =
|
||||||
max_t(__int64_t, fixlen - bmv->bmv_offset, 0);
|
max_t(int64_t, fixlen - bmv->bmv_offset, 0);
|
||||||
} else if (bmv->bmv_length == 0) {
|
} else if (bmv->bmv_length == 0) {
|
||||||
bmv->bmv_entries = 0;
|
bmv->bmv_entries = 0;
|
||||||
return 0;
|
return 0;
|
||||||
@ -742,7 +767,7 @@ xfs_getbmap(
|
|||||||
out[cur_ext].bmv_offset +
|
out[cur_ext].bmv_offset +
|
||||||
out[cur_ext].bmv_length;
|
out[cur_ext].bmv_length;
|
||||||
bmv->bmv_length =
|
bmv->bmv_length =
|
||||||
max_t(__int64_t, 0, bmvend - bmv->bmv_offset);
|
max_t(int64_t, 0, bmvend - bmv->bmv_offset);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* In case we don't want to return the hole,
|
* In case we don't want to return the hole,
|
||||||
@ -1617,7 +1642,7 @@ xfs_swap_extents_check_format(
|
|||||||
* extent format...
|
* extent format...
|
||||||
*/
|
*/
|
||||||
if (tip->i_d.di_format == XFS_DINODE_FMT_BTREE) {
|
if (tip->i_d.di_format == XFS_DINODE_FMT_BTREE) {
|
||||||
if (XFS_IFORK_BOFF(ip) &&
|
if (XFS_IFORK_Q(ip) &&
|
||||||
XFS_BMAP_BMDR_SPACE(tip->i_df.if_broot) > XFS_IFORK_BOFF(ip))
|
XFS_BMAP_BMDR_SPACE(tip->i_df.if_broot) > XFS_IFORK_BOFF(ip))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
if (XFS_IFORK_NEXTENTS(tip, XFS_DATA_FORK) <=
|
if (XFS_IFORK_NEXTENTS(tip, XFS_DATA_FORK) <=
|
||||||
@ -1627,7 +1652,7 @@ xfs_swap_extents_check_format(
|
|||||||
|
|
||||||
/* Reciprocal target->temp btree format checks */
|
/* Reciprocal target->temp btree format checks */
|
||||||
if (ip->i_d.di_format == XFS_DINODE_FMT_BTREE) {
|
if (ip->i_d.di_format == XFS_DINODE_FMT_BTREE) {
|
||||||
if (XFS_IFORK_BOFF(tip) &&
|
if (XFS_IFORK_Q(tip) &&
|
||||||
XFS_BMAP_BMDR_SPACE(ip->i_df.if_broot) > XFS_IFORK_BOFF(tip))
|
XFS_BMAP_BMDR_SPACE(ip->i_df.if_broot) > XFS_IFORK_BOFF(tip))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
if (XFS_IFORK_NEXTENTS(ip, XFS_DATA_FORK) <=
|
if (XFS_IFORK_NEXTENTS(ip, XFS_DATA_FORK) <=
|
||||||
@ -1676,7 +1701,7 @@ xfs_swap_extent_rmap(
|
|||||||
xfs_filblks_t ilen;
|
xfs_filblks_t ilen;
|
||||||
xfs_filblks_t rlen;
|
xfs_filblks_t rlen;
|
||||||
int nimaps;
|
int nimaps;
|
||||||
__uint64_t tip_flags2;
|
uint64_t tip_flags2;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the source file has shared blocks, we must flag the donor
|
* If the source file has shared blocks, we must flag the donor
|
||||||
@ -1789,10 +1814,11 @@ xfs_swap_extent_forks(
|
|||||||
int *target_log_flags)
|
int *target_log_flags)
|
||||||
{
|
{
|
||||||
struct xfs_ifork tempifp, *ifp, *tifp;
|
struct xfs_ifork tempifp, *ifp, *tifp;
|
||||||
int aforkblks = 0;
|
xfs_filblks_t aforkblks = 0;
|
||||||
int taforkblks = 0;
|
xfs_filblks_t taforkblks = 0;
|
||||||
|
xfs_extnum_t junk;
|
||||||
xfs_extnum_t nextents;
|
xfs_extnum_t nextents;
|
||||||
__uint64_t tmp;
|
uint64_t tmp;
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1800,14 +1826,14 @@ xfs_swap_extent_forks(
|
|||||||
*/
|
*/
|
||||||
if ( ((XFS_IFORK_Q(ip) != 0) && (ip->i_d.di_anextents > 0)) &&
|
if ( ((XFS_IFORK_Q(ip) != 0) && (ip->i_d.di_anextents > 0)) &&
|
||||||
(ip->i_d.di_aformat != XFS_DINODE_FMT_LOCAL)) {
|
(ip->i_d.di_aformat != XFS_DINODE_FMT_LOCAL)) {
|
||||||
error = xfs_bmap_count_blocks(tp, ip, XFS_ATTR_FORK,
|
error = xfs_bmap_count_blocks(tp, ip, XFS_ATTR_FORK, &junk,
|
||||||
&aforkblks);
|
&aforkblks);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
if ( ((XFS_IFORK_Q(tip) != 0) && (tip->i_d.di_anextents > 0)) &&
|
if ( ((XFS_IFORK_Q(tip) != 0) && (tip->i_d.di_anextents > 0)) &&
|
||||||
(tip->i_d.di_aformat != XFS_DINODE_FMT_LOCAL)) {
|
(tip->i_d.di_aformat != XFS_DINODE_FMT_LOCAL)) {
|
||||||
error = xfs_bmap_count_blocks(tp, tip, XFS_ATTR_FORK,
|
error = xfs_bmap_count_blocks(tp, tip, XFS_ATTR_FORK, &junk,
|
||||||
&taforkblks);
|
&taforkblks);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
@ -1850,15 +1876,15 @@ xfs_swap_extent_forks(
|
|||||||
/*
|
/*
|
||||||
* Fix the on-disk inode values
|
* Fix the on-disk inode values
|
||||||
*/
|
*/
|
||||||
tmp = (__uint64_t)ip->i_d.di_nblocks;
|
tmp = (uint64_t)ip->i_d.di_nblocks;
|
||||||
ip->i_d.di_nblocks = tip->i_d.di_nblocks - taforkblks + aforkblks;
|
ip->i_d.di_nblocks = tip->i_d.di_nblocks - taforkblks + aforkblks;
|
||||||
tip->i_d.di_nblocks = tmp + taforkblks - aforkblks;
|
tip->i_d.di_nblocks = tmp + taforkblks - aforkblks;
|
||||||
|
|
||||||
tmp = (__uint64_t) ip->i_d.di_nextents;
|
tmp = (uint64_t) ip->i_d.di_nextents;
|
||||||
ip->i_d.di_nextents = tip->i_d.di_nextents;
|
ip->i_d.di_nextents = tip->i_d.di_nextents;
|
||||||
tip->i_d.di_nextents = tmp;
|
tip->i_d.di_nextents = tmp;
|
||||||
|
|
||||||
tmp = (__uint64_t) ip->i_d.di_format;
|
tmp = (uint64_t) ip->i_d.di_format;
|
||||||
ip->i_d.di_format = tip->i_d.di_format;
|
ip->i_d.di_format = tip->i_d.di_format;
|
||||||
tip->i_d.di_format = tmp;
|
tip->i_d.di_format = tmp;
|
||||||
|
|
||||||
@ -1927,7 +1953,7 @@ xfs_swap_extents(
|
|||||||
int error = 0;
|
int error = 0;
|
||||||
int lock_flags;
|
int lock_flags;
|
||||||
struct xfs_ifork *cowfp;
|
struct xfs_ifork *cowfp;
|
||||||
__uint64_t f;
|
uint64_t f;
|
||||||
int resblks;
|
int resblks;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -70,4 +70,8 @@ int xfs_swap_extents(struct xfs_inode *ip, struct xfs_inode *tip,
|
|||||||
|
|
||||||
xfs_daddr_t xfs_fsb_to_db(struct xfs_inode *ip, xfs_fsblock_t fsb);
|
xfs_daddr_t xfs_fsb_to_db(struct xfs_inode *ip, xfs_fsblock_t fsb);
|
||||||
|
|
||||||
|
int xfs_bmap_count_blocks(struct xfs_trans *tp, struct xfs_inode *ip,
|
||||||
|
int whichfork, xfs_extnum_t *nextents,
|
||||||
|
xfs_filblks_t *count);
|
||||||
|
|
||||||
#endif /* __XFS_BMAP_UTIL_H__ */
|
#endif /* __XFS_BMAP_UTIL_H__ */
|
||||||
|
@ -1194,7 +1194,7 @@ xfs_buf_ioerror_alert(
|
|||||||
{
|
{
|
||||||
xfs_alert(bp->b_target->bt_mount,
|
xfs_alert(bp->b_target->bt_mount,
|
||||||
"metadata I/O error: block 0x%llx (\"%s\") error %d numblks %d",
|
"metadata I/O error: block 0x%llx (\"%s\") error %d numblks %d",
|
||||||
(__uint64_t)XFS_BUF_ADDR(bp), func, -bp->b_error, bp->b_length);
|
(uint64_t)XFS_BUF_ADDR(bp), func, -bp->b_error, bp->b_length);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
@ -2050,6 +2050,66 @@ xfs_buf_delwri_submit(
|
|||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Push a single buffer on a delwri queue.
|
||||||
|
*
|
||||||
|
* The purpose of this function is to submit a single buffer of a delwri queue
|
||||||
|
* and return with the buffer still on the original queue. The waiting delwri
|
||||||
|
* buffer submission infrastructure guarantees transfer of the delwri queue
|
||||||
|
* buffer reference to a temporary wait list. We reuse this infrastructure to
|
||||||
|
* transfer the buffer back to the original queue.
|
||||||
|
*
|
||||||
|
* Note the buffer transitions from the queued state, to the submitted and wait
|
||||||
|
* listed state and back to the queued state during this call. The buffer
|
||||||
|
* locking and queue management logic between _delwri_pushbuf() and
|
||||||
|
* _delwri_queue() guarantee that the buffer cannot be queued to another list
|
||||||
|
* before returning.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
xfs_buf_delwri_pushbuf(
|
||||||
|
struct xfs_buf *bp,
|
||||||
|
struct list_head *buffer_list)
|
||||||
|
{
|
||||||
|
LIST_HEAD (submit_list);
|
||||||
|
int error;
|
||||||
|
|
||||||
|
ASSERT(bp->b_flags & _XBF_DELWRI_Q);
|
||||||
|
|
||||||
|
trace_xfs_buf_delwri_pushbuf(bp, _RET_IP_);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Isolate the buffer to a new local list so we can submit it for I/O
|
||||||
|
* independently from the rest of the original list.
|
||||||
|
*/
|
||||||
|
xfs_buf_lock(bp);
|
||||||
|
list_move(&bp->b_list, &submit_list);
|
||||||
|
xfs_buf_unlock(bp);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Delwri submission clears the DELWRI_Q buffer flag and returns with
|
||||||
|
* the buffer on the wait list with an associated reference. Rather than
|
||||||
|
* bounce the buffer from a local wait list back to the original list
|
||||||
|
* after I/O completion, reuse the original list as the wait list.
|
||||||
|
*/
|
||||||
|
xfs_buf_delwri_submit_buffers(&submit_list, buffer_list);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The buffer is now under I/O and wait listed as during typical delwri
|
||||||
|
* submission. Lock the buffer to wait for I/O completion. Rather than
|
||||||
|
* remove the buffer from the wait list and release the reference, we
|
||||||
|
* want to return with the buffer queued to the original list. The
|
||||||
|
* buffer already sits on the original list with a wait list reference,
|
||||||
|
* however. If we let the queue inherit that wait list reference, all we
|
||||||
|
* need to do is reset the DELWRI_Q flag.
|
||||||
|
*/
|
||||||
|
xfs_buf_lock(bp);
|
||||||
|
error = bp->b_error;
|
||||||
|
bp->b_flags |= _XBF_DELWRI_Q;
|
||||||
|
xfs_buf_unlock(bp);
|
||||||
|
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
int __init
|
int __init
|
||||||
xfs_buf_init(void)
|
xfs_buf_init(void)
|
||||||
{
|
{
|
||||||
|
@ -332,6 +332,7 @@ extern void xfs_buf_delwri_cancel(struct list_head *);
|
|||||||
extern bool xfs_buf_delwri_queue(struct xfs_buf *, struct list_head *);
|
extern bool xfs_buf_delwri_queue(struct xfs_buf *, struct list_head *);
|
||||||
extern int xfs_buf_delwri_submit(struct list_head *);
|
extern int xfs_buf_delwri_submit(struct list_head *);
|
||||||
extern int xfs_buf_delwri_submit_nowait(struct list_head *);
|
extern int xfs_buf_delwri_submit_nowait(struct list_head *);
|
||||||
|
extern int xfs_buf_delwri_pushbuf(struct xfs_buf *, struct list_head *);
|
||||||
|
|
||||||
/* Buffer Daemon Setup Routines */
|
/* Buffer Daemon Setup Routines */
|
||||||
extern int xfs_buf_init(void);
|
extern int xfs_buf_init(void);
|
||||||
|
@ -636,20 +636,23 @@ xfs_buf_item_unlock(
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Clean buffers, by definition, cannot be in the AIL. However, aborted
|
* Clean buffers, by definition, cannot be in the AIL. However, aborted
|
||||||
* buffers may be dirty and hence in the AIL. Therefore if we are
|
* buffers may be in the AIL regardless of dirty state. An aborted
|
||||||
* aborting a buffer and we've just taken the last refernce away, we
|
* transaction that invalidates a buffer already in the AIL may have
|
||||||
* have to check if it is in the AIL before freeing it. We need to free
|
* marked it stale and cleared the dirty state, for example.
|
||||||
* it in this case, because an aborted transaction has already shut the
|
*
|
||||||
* filesystem down and this is the last chance we will have to do so.
|
* Therefore if we are aborting a buffer and we've just taken the last
|
||||||
|
* reference away, we have to check if it is in the AIL before freeing
|
||||||
|
* it. We need to free it in this case, because an aborted transaction
|
||||||
|
* has already shut the filesystem down and this is the last chance we
|
||||||
|
* will have to do so.
|
||||||
*/
|
*/
|
||||||
if (atomic_dec_and_test(&bip->bli_refcount)) {
|
if (atomic_dec_and_test(&bip->bli_refcount)) {
|
||||||
if (clean)
|
if (aborted) {
|
||||||
xfs_buf_item_relse(bp);
|
|
||||||
else if (aborted) {
|
|
||||||
ASSERT(XFS_FORCED_SHUTDOWN(lip->li_mountp));
|
ASSERT(XFS_FORCED_SHUTDOWN(lip->li_mountp));
|
||||||
xfs_trans_ail_remove(lip, SHUTDOWN_LOG_IO_ERROR);
|
xfs_trans_ail_remove(lip, SHUTDOWN_LOG_IO_ERROR);
|
||||||
xfs_buf_item_relse(bp);
|
xfs_buf_item_relse(bp);
|
||||||
}
|
} else if (clean)
|
||||||
|
xfs_buf_item_relse(bp);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(flags & XFS_BLI_HOLD))
|
if (!(flags & XFS_BLI_HOLD))
|
||||||
|
@ -44,7 +44,7 @@ static unsigned char xfs_dir3_filetype_table[] = {
|
|||||||
static unsigned char
|
static unsigned char
|
||||||
xfs_dir3_get_dtype(
|
xfs_dir3_get_dtype(
|
||||||
struct xfs_mount *mp,
|
struct xfs_mount *mp,
|
||||||
__uint8_t filetype)
|
uint8_t filetype)
|
||||||
{
|
{
|
||||||
if (!xfs_sb_version_hasftype(&mp->m_sb))
|
if (!xfs_sb_version_hasftype(&mp->m_sb))
|
||||||
return DT_UNKNOWN;
|
return DT_UNKNOWN;
|
||||||
@ -117,7 +117,7 @@ xfs_dir2_sf_getdents(
|
|||||||
*/
|
*/
|
||||||
sfep = xfs_dir2_sf_firstentry(sfp);
|
sfep = xfs_dir2_sf_firstentry(sfp);
|
||||||
for (i = 0; i < sfp->count; i++) {
|
for (i = 0; i < sfp->count; i++) {
|
||||||
__uint8_t filetype;
|
uint8_t filetype;
|
||||||
|
|
||||||
off = xfs_dir2_db_off_to_dataptr(geo, geo->datablk,
|
off = xfs_dir2_db_off_to_dataptr(geo, geo->datablk,
|
||||||
xfs_dir2_sf_get_offset(sfep));
|
xfs_dir2_sf_get_offset(sfep));
|
||||||
@ -170,7 +170,7 @@ xfs_dir2_block_getdents(
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
lock_mode = xfs_ilock_data_map_shared(dp);
|
lock_mode = xfs_ilock_data_map_shared(dp);
|
||||||
error = xfs_dir3_block_read(NULL, dp, &bp);
|
error = xfs_dir3_block_read(args->trans, dp, &bp);
|
||||||
xfs_iunlock(dp, lock_mode);
|
xfs_iunlock(dp, lock_mode);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
@ -194,7 +194,7 @@ xfs_dir2_block_getdents(
|
|||||||
* Each object is a real entry (dep) or an unused one (dup).
|
* Each object is a real entry (dep) or an unused one (dup).
|
||||||
*/
|
*/
|
||||||
while (ptr < endptr) {
|
while (ptr < endptr) {
|
||||||
__uint8_t filetype;
|
uint8_t filetype;
|
||||||
|
|
||||||
dup = (xfs_dir2_data_unused_t *)ptr;
|
dup = (xfs_dir2_data_unused_t *)ptr;
|
||||||
/*
|
/*
|
||||||
@ -228,7 +228,7 @@ xfs_dir2_block_getdents(
|
|||||||
if (!dir_emit(ctx, (char *)dep->name, dep->namelen,
|
if (!dir_emit(ctx, (char *)dep->name, dep->namelen,
|
||||||
be64_to_cpu(dep->inumber),
|
be64_to_cpu(dep->inumber),
|
||||||
xfs_dir3_get_dtype(dp->i_mount, filetype))) {
|
xfs_dir3_get_dtype(dp->i_mount, filetype))) {
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(args->trans, bp);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -239,218 +239,104 @@ xfs_dir2_block_getdents(
|
|||||||
*/
|
*/
|
||||||
ctx->pos = xfs_dir2_db_off_to_dataptr(geo, geo->datablk + 1, 0) &
|
ctx->pos = xfs_dir2_db_off_to_dataptr(geo, geo->datablk + 1, 0) &
|
||||||
0x7fffffff;
|
0x7fffffff;
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(args->trans, bp);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct xfs_dir2_leaf_map_info {
|
/*
|
||||||
xfs_extlen_t map_blocks; /* number of fsbs in map */
|
* Read a directory block and initiate readahead for blocks beyond that.
|
||||||
xfs_dablk_t map_off; /* last mapped file offset */
|
* We maintain a sliding readahead window of the remaining space in the
|
||||||
int map_size; /* total entries in *map */
|
* buffer rounded up to the nearest block.
|
||||||
int map_valid; /* valid entries in *map */
|
*/
|
||||||
int nmap; /* mappings to ask xfs_bmapi */
|
|
||||||
xfs_dir2_db_t curdb; /* db for current block */
|
|
||||||
int ra_current; /* number of read-ahead blks */
|
|
||||||
int ra_index; /* *map index for read-ahead */
|
|
||||||
int ra_offset; /* map entry offset for ra */
|
|
||||||
int ra_want; /* readahead count wanted */
|
|
||||||
struct xfs_bmbt_irec map[]; /* map vector for blocks */
|
|
||||||
};
|
|
||||||
|
|
||||||
STATIC int
|
STATIC int
|
||||||
xfs_dir2_leaf_readbuf(
|
xfs_dir2_leaf_readbuf(
|
||||||
struct xfs_da_args *args,
|
struct xfs_da_args *args,
|
||||||
size_t bufsize,
|
size_t bufsize,
|
||||||
struct xfs_dir2_leaf_map_info *mip,
|
xfs_dir2_off_t *cur_off,
|
||||||
xfs_dir2_off_t *curoff,
|
xfs_dablk_t *ra_blk,
|
||||||
struct xfs_buf **bpp,
|
struct xfs_buf **bpp)
|
||||||
bool trim_map)
|
|
||||||
{
|
{
|
||||||
struct xfs_inode *dp = args->dp;
|
struct xfs_inode *dp = args->dp;
|
||||||
struct xfs_buf *bp = NULL;
|
struct xfs_buf *bp = NULL;
|
||||||
struct xfs_bmbt_irec *map = mip->map;
|
|
||||||
struct blk_plug plug;
|
|
||||||
int error = 0;
|
|
||||||
int length;
|
|
||||||
int i;
|
|
||||||
int j;
|
|
||||||
struct xfs_da_geometry *geo = args->geo;
|
struct xfs_da_geometry *geo = args->geo;
|
||||||
|
struct xfs_ifork *ifp = XFS_IFORK_PTR(dp, XFS_DATA_FORK);
|
||||||
|
struct xfs_bmbt_irec map;
|
||||||
|
struct blk_plug plug;
|
||||||
|
xfs_dir2_off_t new_off;
|
||||||
|
xfs_dablk_t next_ra;
|
||||||
|
xfs_dablk_t map_off;
|
||||||
|
xfs_dablk_t last_da;
|
||||||
|
xfs_extnum_t idx;
|
||||||
|
int ra_want;
|
||||||
|
int error = 0;
|
||||||
|
|
||||||
/*
|
if (!(ifp->if_flags & XFS_IFEXTENTS)) {
|
||||||
* If the caller just finished processing a buffer, it will tell us
|
error = xfs_iread_extents(args->trans, dp, XFS_DATA_FORK);
|
||||||
* we need to trim that block out of the mapping now it is done.
|
|
||||||
*/
|
|
||||||
if (trim_map) {
|
|
||||||
mip->map_blocks -= geo->fsbcount;
|
|
||||||
/*
|
|
||||||
* Loop to get rid of the extents for the
|
|
||||||
* directory block.
|
|
||||||
*/
|
|
||||||
for (i = geo->fsbcount; i > 0; ) {
|
|
||||||
j = min_t(int, map->br_blockcount, i);
|
|
||||||
map->br_blockcount -= j;
|
|
||||||
map->br_startblock += j;
|
|
||||||
map->br_startoff += j;
|
|
||||||
/*
|
|
||||||
* If mapping is done, pitch it from
|
|
||||||
* the table.
|
|
||||||
*/
|
|
||||||
if (!map->br_blockcount && --mip->map_valid)
|
|
||||||
memmove(&map[0], &map[1],
|
|
||||||
sizeof(map[0]) * mip->map_valid);
|
|
||||||
i -= j;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Recalculate the readahead blocks wanted.
|
|
||||||
*/
|
|
||||||
mip->ra_want = howmany(bufsize + geo->blksize, (1 << geo->fsblog)) - 1;
|
|
||||||
ASSERT(mip->ra_want >= 0);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* If we don't have as many as we want, and we haven't
|
|
||||||
* run out of data blocks, get some more mappings.
|
|
||||||
*/
|
|
||||||
if (1 + mip->ra_want > mip->map_blocks &&
|
|
||||||
mip->map_off < xfs_dir2_byte_to_da(geo, XFS_DIR2_LEAF_OFFSET)) {
|
|
||||||
/*
|
|
||||||
* Get more bmaps, fill in after the ones
|
|
||||||
* we already have in the table.
|
|
||||||
*/
|
|
||||||
mip->nmap = mip->map_size - mip->map_valid;
|
|
||||||
error = xfs_bmapi_read(dp, mip->map_off,
|
|
||||||
xfs_dir2_byte_to_da(geo, XFS_DIR2_LEAF_OFFSET) -
|
|
||||||
mip->map_off,
|
|
||||||
&map[mip->map_valid], &mip->nmap, 0);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Don't know if we should ignore this or try to return an
|
|
||||||
* error. The trouble with returning errors is that readdir
|
|
||||||
* will just stop without actually passing the error through.
|
|
||||||
*/
|
|
||||||
if (error)
|
if (error)
|
||||||
goto out; /* XXX */
|
goto out;
|
||||||
|
|
||||||
/*
|
|
||||||
* If we got all the mappings we asked for, set the final map
|
|
||||||
* offset based on the last bmap value received. Otherwise,
|
|
||||||
* we've reached the end.
|
|
||||||
*/
|
|
||||||
if (mip->nmap == mip->map_size - mip->map_valid) {
|
|
||||||
i = mip->map_valid + mip->nmap - 1;
|
|
||||||
mip->map_off = map[i].br_startoff + map[i].br_blockcount;
|
|
||||||
} else
|
|
||||||
mip->map_off = xfs_dir2_byte_to_da(geo,
|
|
||||||
XFS_DIR2_LEAF_OFFSET);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Look for holes in the mapping, and eliminate them. Count up
|
|
||||||
* the valid blocks.
|
|
||||||
*/
|
|
||||||
for (i = mip->map_valid; i < mip->map_valid + mip->nmap; ) {
|
|
||||||
if (map[i].br_startblock == HOLESTARTBLOCK) {
|
|
||||||
mip->nmap--;
|
|
||||||
length = mip->map_valid + mip->nmap - i;
|
|
||||||
if (length)
|
|
||||||
memmove(&map[i], &map[i + 1],
|
|
||||||
sizeof(map[i]) * length);
|
|
||||||
} else {
|
|
||||||
mip->map_blocks += map[i].br_blockcount;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
mip->map_valid += mip->nmap;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* No valid mappings, so no more data blocks.
|
* Look for mapped directory blocks at or above the current offset.
|
||||||
|
* Truncate down to the nearest directory block to start the scanning
|
||||||
|
* operation.
|
||||||
*/
|
*/
|
||||||
if (!mip->map_valid) {
|
last_da = xfs_dir2_byte_to_da(geo, XFS_DIR2_LEAF_OFFSET);
|
||||||
*curoff = xfs_dir2_da_to_byte(geo, mip->map_off);
|
map_off = xfs_dir2_db_to_da(geo, xfs_dir2_byte_to_db(geo, *cur_off));
|
||||||
|
if (!xfs_iext_lookup_extent(dp, ifp, map_off, &idx, &map))
|
||||||
goto out;
|
goto out;
|
||||||
}
|
if (map.br_startoff >= last_da)
|
||||||
|
goto out;
|
||||||
|
xfs_trim_extent(&map, map_off, last_da - map_off);
|
||||||
|
|
||||||
/*
|
/* Read the directory block of that first mapping. */
|
||||||
* Read the directory block starting at the first mapping.
|
new_off = xfs_dir2_da_to_byte(geo, map.br_startoff);
|
||||||
*/
|
if (new_off > *cur_off)
|
||||||
mip->curdb = xfs_dir2_da_to_db(geo, map->br_startoff);
|
*cur_off = new_off;
|
||||||
error = xfs_dir3_data_read(NULL, dp, map->br_startoff,
|
error = xfs_dir3_data_read(args->trans, dp, map.br_startoff, -1, &bp);
|
||||||
map->br_blockcount >= geo->fsbcount ?
|
|
||||||
XFS_FSB_TO_DADDR(dp->i_mount, map->br_startblock) :
|
|
||||||
-1, &bp);
|
|
||||||
/*
|
|
||||||
* Should just skip over the data block instead of giving up.
|
|
||||||
*/
|
|
||||||
if (error)
|
if (error)
|
||||||
goto out; /* XXX */
|
goto out;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Adjust the current amount of read-ahead: we just read a block that
|
* Start readahead for the next bufsize's worth of dir data blocks.
|
||||||
* was previously ra.
|
* We may have already issued readahead for some of that range;
|
||||||
|
* ra_blk tracks the last block we tried to read(ahead).
|
||||||
*/
|
*/
|
||||||
if (mip->ra_current)
|
ra_want = howmany(bufsize + geo->blksize, (1 << geo->fsblog));
|
||||||
mip->ra_current -= geo->fsbcount;
|
if (*ra_blk >= last_da)
|
||||||
|
goto out;
|
||||||
|
else if (*ra_blk == 0)
|
||||||
|
*ra_blk = map.br_startoff;
|
||||||
|
next_ra = map.br_startoff + geo->fsbcount;
|
||||||
|
if (next_ra >= last_da)
|
||||||
|
goto out_no_ra;
|
||||||
|
if (map.br_blockcount < geo->fsbcount &&
|
||||||
|
!xfs_iext_get_extent(ifp, ++idx, &map))
|
||||||
|
goto out_no_ra;
|
||||||
|
if (map.br_startoff >= last_da)
|
||||||
|
goto out_no_ra;
|
||||||
|
xfs_trim_extent(&map, next_ra, last_da - next_ra);
|
||||||
|
|
||||||
/*
|
/* Start ra for each dir (not fs) block that has a mapping. */
|
||||||
* Do we need more readahead?
|
|
||||||
* Each loop tries to process 1 full dir blk; last may be partial.
|
|
||||||
*/
|
|
||||||
blk_start_plug(&plug);
|
blk_start_plug(&plug);
|
||||||
for (mip->ra_index = mip->ra_offset = i = 0;
|
while (ra_want > 0) {
|
||||||
mip->ra_want > mip->ra_current && i < mip->map_blocks;
|
next_ra = roundup((xfs_dablk_t)map.br_startoff, geo->fsbcount);
|
||||||
i += geo->fsbcount) {
|
while (ra_want > 0 &&
|
||||||
ASSERT(mip->ra_index < mip->map_valid);
|
next_ra < map.br_startoff + map.br_blockcount) {
|
||||||
/*
|
if (next_ra >= last_da) {
|
||||||
* Read-ahead a contiguous directory block.
|
*ra_blk = last_da;
|
||||||
*/
|
break;
|
||||||
if (i > mip->ra_current &&
|
|
||||||
(map[mip->ra_index].br_blockcount - mip->ra_offset) >=
|
|
||||||
geo->fsbcount) {
|
|
||||||
xfs_dir3_data_readahead(dp,
|
|
||||||
map[mip->ra_index].br_startoff + mip->ra_offset,
|
|
||||||
XFS_FSB_TO_DADDR(dp->i_mount,
|
|
||||||
map[mip->ra_index].br_startblock +
|
|
||||||
mip->ra_offset));
|
|
||||||
mip->ra_current = i;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Read-ahead a non-contiguous directory block. This doesn't
|
|
||||||
* use our mapping, but this is a very rare case.
|
|
||||||
*/
|
|
||||||
else if (i > mip->ra_current) {
|
|
||||||
xfs_dir3_data_readahead(dp,
|
|
||||||
map[mip->ra_index].br_startoff +
|
|
||||||
mip->ra_offset, -1);
|
|
||||||
mip->ra_current = i;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Advance offset through the mapping table, processing a full
|
|
||||||
* dir block even if it is fragmented into several extents.
|
|
||||||
* But stop if we have consumed all valid mappings, even if
|
|
||||||
* it's not yet a full directory block.
|
|
||||||
*/
|
|
||||||
for (j = 0;
|
|
||||||
j < geo->fsbcount && mip->ra_index < mip->map_valid;
|
|
||||||
j += length ) {
|
|
||||||
/*
|
|
||||||
* The rest of this extent but not more than a dir
|
|
||||||
* block.
|
|
||||||
*/
|
|
||||||
length = min_t(int, geo->fsbcount - j,
|
|
||||||
map[mip->ra_index].br_blockcount -
|
|
||||||
mip->ra_offset);
|
|
||||||
mip->ra_offset += length;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Advance to the next mapping if this one is used up.
|
|
||||||
*/
|
|
||||||
if (mip->ra_offset == map[mip->ra_index].br_blockcount) {
|
|
||||||
mip->ra_offset = 0;
|
|
||||||
mip->ra_index++;
|
|
||||||
}
|
}
|
||||||
|
if (next_ra > *ra_blk) {
|
||||||
|
xfs_dir3_data_readahead(dp, next_ra, -2);
|
||||||
|
*ra_blk = next_ra;
|
||||||
|
}
|
||||||
|
ra_want -= geo->fsbcount;
|
||||||
|
next_ra += geo->fsbcount;
|
||||||
|
}
|
||||||
|
if (!xfs_iext_get_extent(ifp, ++idx, &map)) {
|
||||||
|
*ra_blk = last_da;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
blk_finish_plug(&plug);
|
blk_finish_plug(&plug);
|
||||||
@ -458,6 +344,9 @@ xfs_dir2_leaf_readbuf(
|
|||||||
out:
|
out:
|
||||||
*bpp = bp;
|
*bpp = bp;
|
||||||
return error;
|
return error;
|
||||||
|
out_no_ra:
|
||||||
|
*ra_blk = last_da;
|
||||||
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -475,14 +364,14 @@ xfs_dir2_leaf_getdents(
|
|||||||
xfs_dir2_data_hdr_t *hdr; /* data block header */
|
xfs_dir2_data_hdr_t *hdr; /* data block header */
|
||||||
xfs_dir2_data_entry_t *dep; /* data entry */
|
xfs_dir2_data_entry_t *dep; /* data entry */
|
||||||
xfs_dir2_data_unused_t *dup; /* unused entry */
|
xfs_dir2_data_unused_t *dup; /* unused entry */
|
||||||
int error = 0; /* error return value */
|
char *ptr = NULL; /* pointer to current data */
|
||||||
|
struct xfs_da_geometry *geo = args->geo;
|
||||||
|
xfs_dablk_t rablk = 0; /* current readahead block */
|
||||||
|
xfs_dir2_off_t curoff; /* current overall offset */
|
||||||
int length; /* temporary length value */
|
int length; /* temporary length value */
|
||||||
int byteoff; /* offset in current block */
|
int byteoff; /* offset in current block */
|
||||||
xfs_dir2_off_t curoff; /* current overall offset */
|
int lock_mode;
|
||||||
xfs_dir2_off_t newoff; /* new curoff after new blk */
|
int error = 0; /* error return value */
|
||||||
char *ptr = NULL; /* pointer to current data */
|
|
||||||
struct xfs_dir2_leaf_map_info *map_info;
|
|
||||||
struct xfs_da_geometry *geo = args->geo;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the offset is at or past the largest allowed value,
|
* If the offset is at or past the largest allowed value,
|
||||||
@ -491,74 +380,36 @@ xfs_dir2_leaf_getdents(
|
|||||||
if (ctx->pos >= XFS_DIR2_MAX_DATAPTR)
|
if (ctx->pos >= XFS_DIR2_MAX_DATAPTR)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/*
|
|
||||||
* Set up to bmap a number of blocks based on the caller's
|
|
||||||
* buffer size, the directory block size, and the filesystem
|
|
||||||
* block size.
|
|
||||||
*/
|
|
||||||
length = howmany(bufsize + geo->blksize, (1 << geo->fsblog));
|
|
||||||
map_info = kmem_zalloc(offsetof(struct xfs_dir2_leaf_map_info, map) +
|
|
||||||
(length * sizeof(struct xfs_bmbt_irec)),
|
|
||||||
KM_SLEEP | KM_NOFS);
|
|
||||||
map_info->map_size = length;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Inside the loop we keep the main offset value as a byte offset
|
* Inside the loop we keep the main offset value as a byte offset
|
||||||
* in the directory file.
|
* in the directory file.
|
||||||
*/
|
*/
|
||||||
curoff = xfs_dir2_dataptr_to_byte(ctx->pos);
|
curoff = xfs_dir2_dataptr_to_byte(ctx->pos);
|
||||||
|
|
||||||
/*
|
|
||||||
* Force this conversion through db so we truncate the offset
|
|
||||||
* down to get the start of the data block.
|
|
||||||
*/
|
|
||||||
map_info->map_off = xfs_dir2_db_to_da(geo,
|
|
||||||
xfs_dir2_byte_to_db(geo, curoff));
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Loop over directory entries until we reach the end offset.
|
* Loop over directory entries until we reach the end offset.
|
||||||
* Get more blocks and readahead as necessary.
|
* Get more blocks and readahead as necessary.
|
||||||
*/
|
*/
|
||||||
while (curoff < XFS_DIR2_LEAF_OFFSET) {
|
while (curoff < XFS_DIR2_LEAF_OFFSET) {
|
||||||
__uint8_t filetype;
|
uint8_t filetype;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If we have no buffer, or we're off the end of the
|
* If we have no buffer, or we're off the end of the
|
||||||
* current buffer, need to get another one.
|
* current buffer, need to get another one.
|
||||||
*/
|
*/
|
||||||
if (!bp || ptr >= (char *)bp->b_addr + geo->blksize) {
|
if (!bp || ptr >= (char *)bp->b_addr + geo->blksize) {
|
||||||
int lock_mode;
|
|
||||||
bool trim_map = false;
|
|
||||||
|
|
||||||
if (bp) {
|
if (bp) {
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(args->trans, bp);
|
||||||
bp = NULL;
|
bp = NULL;
|
||||||
trim_map = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
lock_mode = xfs_ilock_data_map_shared(dp);
|
lock_mode = xfs_ilock_data_map_shared(dp);
|
||||||
error = xfs_dir2_leaf_readbuf(args, bufsize, map_info,
|
error = xfs_dir2_leaf_readbuf(args, bufsize, &curoff,
|
||||||
&curoff, &bp, trim_map);
|
&rablk, &bp);
|
||||||
xfs_iunlock(dp, lock_mode);
|
xfs_iunlock(dp, lock_mode);
|
||||||
if (error || !map_info->map_valid)
|
if (error || !bp)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/*
|
|
||||||
* Having done a read, we need to set a new offset.
|
|
||||||
*/
|
|
||||||
newoff = xfs_dir2_db_off_to_byte(geo,
|
|
||||||
map_info->curdb, 0);
|
|
||||||
/*
|
|
||||||
* Start of the current block.
|
|
||||||
*/
|
|
||||||
if (curoff < newoff)
|
|
||||||
curoff = newoff;
|
|
||||||
/*
|
|
||||||
* Make sure we're in the right block.
|
|
||||||
*/
|
|
||||||
else if (curoff > newoff)
|
|
||||||
ASSERT(xfs_dir2_byte_to_db(geo, curoff) ==
|
|
||||||
map_info->curdb);
|
|
||||||
hdr = bp->b_addr;
|
hdr = bp->b_addr;
|
||||||
xfs_dir3_data_check(dp, bp);
|
xfs_dir3_data_check(dp, bp);
|
||||||
/*
|
/*
|
||||||
@ -643,17 +494,22 @@ xfs_dir2_leaf_getdents(
|
|||||||
ctx->pos = XFS_DIR2_MAX_DATAPTR & 0x7fffffff;
|
ctx->pos = XFS_DIR2_MAX_DATAPTR & 0x7fffffff;
|
||||||
else
|
else
|
||||||
ctx->pos = xfs_dir2_byte_to_dataptr(curoff) & 0x7fffffff;
|
ctx->pos = xfs_dir2_byte_to_dataptr(curoff) & 0x7fffffff;
|
||||||
kmem_free(map_info);
|
|
||||||
if (bp)
|
if (bp)
|
||||||
xfs_trans_brelse(NULL, bp);
|
xfs_trans_brelse(args->trans, bp);
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read a directory.
|
* Read a directory.
|
||||||
|
*
|
||||||
|
* If supplied, the transaction collects locked dir buffers to avoid
|
||||||
|
* nested buffer deadlocks. This function does not dirty the
|
||||||
|
* transaction. The caller should ensure that the inode is locked
|
||||||
|
* before calling this function.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
xfs_readdir(
|
xfs_readdir(
|
||||||
|
struct xfs_trans *tp,
|
||||||
struct xfs_inode *dp,
|
struct xfs_inode *dp,
|
||||||
struct dir_context *ctx,
|
struct dir_context *ctx,
|
||||||
size_t bufsize)
|
size_t bufsize)
|
||||||
@ -672,6 +528,7 @@ xfs_readdir(
|
|||||||
|
|
||||||
args.dp = dp;
|
args.dp = dp;
|
||||||
args.geo = dp->i_mount->m_dir_geo;
|
args.geo = dp->i_mount->m_dir_geo;
|
||||||
|
args.trans = tp;
|
||||||
|
|
||||||
if (dp->i_d.di_format == XFS_DINODE_FMT_LOCAL)
|
if (dp->i_d.di_format == XFS_DINODE_FMT_LOCAL)
|
||||||
rval = xfs_dir2_sf_getdents(&args, ctx);
|
rval = xfs_dir2_sf_getdents(&args, ctx);
|
||||||
|
@ -39,7 +39,7 @@ xfs_trim_extents(
|
|||||||
xfs_daddr_t start,
|
xfs_daddr_t start,
|
||||||
xfs_daddr_t end,
|
xfs_daddr_t end,
|
||||||
xfs_daddr_t minlen,
|
xfs_daddr_t minlen,
|
||||||
__uint64_t *blocks_trimmed)
|
uint64_t *blocks_trimmed)
|
||||||
{
|
{
|
||||||
struct block_device *bdev = mp->m_ddev_targp->bt_bdev;
|
struct block_device *bdev = mp->m_ddev_targp->bt_bdev;
|
||||||
struct xfs_btree_cur *cur;
|
struct xfs_btree_cur *cur;
|
||||||
@ -166,7 +166,7 @@ xfs_ioc_trim(
|
|||||||
struct fstrim_range range;
|
struct fstrim_range range;
|
||||||
xfs_daddr_t start, end, minlen;
|
xfs_daddr_t start, end, minlen;
|
||||||
xfs_agnumber_t start_agno, end_agno, agno;
|
xfs_agnumber_t start_agno, end_agno, agno;
|
||||||
__uint64_t blocks_trimmed = 0;
|
uint64_t blocks_trimmed = 0;
|
||||||
int error, last_error = 0;
|
int error, last_error = 0;
|
||||||
|
|
||||||
if (!capable(CAP_SYS_ADMIN))
|
if (!capable(CAP_SYS_ADMIN))
|
||||||
|
@ -276,7 +276,7 @@ xfs_qm_init_dquot_blk(
|
|||||||
void
|
void
|
||||||
xfs_dquot_set_prealloc_limits(struct xfs_dquot *dqp)
|
xfs_dquot_set_prealloc_limits(struct xfs_dquot *dqp)
|
||||||
{
|
{
|
||||||
__uint64_t space;
|
uint64_t space;
|
||||||
|
|
||||||
dqp->q_prealloc_hi_wmark = be64_to_cpu(dqp->q_core.d_blk_hardlimit);
|
dqp->q_prealloc_hi_wmark = be64_to_cpu(dqp->q_core.d_blk_hardlimit);
|
||||||
dqp->q_prealloc_lo_wmark = be64_to_cpu(dqp->q_core.d_blk_softlimit);
|
dqp->q_prealloc_lo_wmark = be64_to_cpu(dqp->q_core.d_blk_softlimit);
|
||||||
@ -472,18 +472,23 @@ xfs_qm_dqtobp(
|
|||||||
struct xfs_mount *mp = dqp->q_mount;
|
struct xfs_mount *mp = dqp->q_mount;
|
||||||
xfs_dqid_t id = be32_to_cpu(dqp->q_core.d_id);
|
xfs_dqid_t id = be32_to_cpu(dqp->q_core.d_id);
|
||||||
struct xfs_trans *tp = (tpp ? *tpp : NULL);
|
struct xfs_trans *tp = (tpp ? *tpp : NULL);
|
||||||
uint lock_mode;
|
uint lock_mode = 0;
|
||||||
|
|
||||||
quotip = xfs_quota_inode(dqp->q_mount, dqp->dq_flags);
|
quotip = xfs_quota_inode(dqp->q_mount, dqp->dq_flags);
|
||||||
dqp->q_fileoffset = (xfs_fileoff_t)id / mp->m_quotainfo->qi_dqperchunk;
|
dqp->q_fileoffset = (xfs_fileoff_t)id / mp->m_quotainfo->qi_dqperchunk;
|
||||||
|
|
||||||
lock_mode = xfs_ilock_data_map_shared(quotip);
|
ASSERT(!(flags & XFS_QMOPT_NOLOCK) ||
|
||||||
|
xfs_isilocked(quotip, XFS_ILOCK_SHARED) ||
|
||||||
|
xfs_isilocked(quotip, XFS_ILOCK_EXCL));
|
||||||
|
if (!(flags & XFS_QMOPT_NOLOCK))
|
||||||
|
lock_mode = xfs_ilock_data_map_shared(quotip);
|
||||||
if (!xfs_this_quota_on(dqp->q_mount, dqp->dq_flags)) {
|
if (!xfs_this_quota_on(dqp->q_mount, dqp->dq_flags)) {
|
||||||
/*
|
/*
|
||||||
* Return if this type of quotas is turned off while we
|
* Return if this type of quotas is turned off while we
|
||||||
* didn't have the quota inode lock.
|
* didn't have the quota inode lock.
|
||||||
*/
|
*/
|
||||||
xfs_iunlock(quotip, lock_mode);
|
if (lock_mode)
|
||||||
|
xfs_iunlock(quotip, lock_mode);
|
||||||
return -ESRCH;
|
return -ESRCH;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -493,7 +498,8 @@ xfs_qm_dqtobp(
|
|||||||
error = xfs_bmapi_read(quotip, dqp->q_fileoffset,
|
error = xfs_bmapi_read(quotip, dqp->q_fileoffset,
|
||||||
XFS_DQUOT_CLUSTER_SIZE_FSB, &map, &nmaps, 0);
|
XFS_DQUOT_CLUSTER_SIZE_FSB, &map, &nmaps, 0);
|
||||||
|
|
||||||
xfs_iunlock(quotip, lock_mode);
|
if (lock_mode)
|
||||||
|
xfs_iunlock(quotip, lock_mode);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
|
|
||||||
@ -695,21 +701,18 @@ error0:
|
|||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
xfs_dq_get_next_id(
|
xfs_dq_get_next_id(
|
||||||
xfs_mount_t *mp,
|
struct xfs_mount *mp,
|
||||||
uint type,
|
uint type,
|
||||||
xfs_dqid_t *id,
|
xfs_dqid_t *id)
|
||||||
loff_t eof)
|
|
||||||
{
|
{
|
||||||
struct xfs_inode *quotip;
|
struct xfs_inode *quotip = xfs_quota_inode(mp, type);
|
||||||
|
xfs_dqid_t next_id = *id + 1; /* simple advance */
|
||||||
|
uint lock_flags;
|
||||||
|
struct xfs_bmbt_irec got;
|
||||||
|
xfs_extnum_t idx;
|
||||||
xfs_fsblock_t start;
|
xfs_fsblock_t start;
|
||||||
loff_t offset;
|
|
||||||
uint lock;
|
|
||||||
xfs_dqid_t next_id;
|
|
||||||
int error = 0;
|
int error = 0;
|
||||||
|
|
||||||
/* Simple advance */
|
|
||||||
next_id = *id + 1;
|
|
||||||
|
|
||||||
/* If we'd wrap past the max ID, stop */
|
/* If we'd wrap past the max ID, stop */
|
||||||
if (next_id < *id)
|
if (next_id < *id)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
@ -723,23 +726,25 @@ xfs_dq_get_next_id(
|
|||||||
/* Nope, next_id is now past the current chunk, so find the next one */
|
/* Nope, next_id is now past the current chunk, so find the next one */
|
||||||
start = (xfs_fsblock_t)next_id / mp->m_quotainfo->qi_dqperchunk;
|
start = (xfs_fsblock_t)next_id / mp->m_quotainfo->qi_dqperchunk;
|
||||||
|
|
||||||
quotip = xfs_quota_inode(mp, type);
|
lock_flags = xfs_ilock_data_map_shared(quotip);
|
||||||
lock = xfs_ilock_data_map_shared(quotip);
|
if (!(quotip->i_df.if_flags & XFS_IFEXTENTS)) {
|
||||||
|
error = xfs_iread_extents(NULL, quotip, XFS_DATA_FORK);
|
||||||
|
if (error)
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
offset = __xfs_seek_hole_data(VFS_I(quotip), XFS_FSB_TO_B(mp, start),
|
if (xfs_iext_lookup_extent(quotip, "ip->i_df, start, &idx, &got)) {
|
||||||
eof, SEEK_DATA);
|
/* contiguous chunk, bump startoff for the id calculation */
|
||||||
if (offset < 0)
|
if (got.br_startoff < start)
|
||||||
error = offset;
|
got.br_startoff = start;
|
||||||
|
*id = got.br_startoff * mp->m_quotainfo->qi_dqperchunk;
|
||||||
|
} else {
|
||||||
|
error = -ENOENT;
|
||||||
|
}
|
||||||
|
|
||||||
xfs_iunlock(quotip, lock);
|
xfs_iunlock(quotip, lock_flags);
|
||||||
|
|
||||||
/* -ENXIO is essentially "no more data" */
|
return error;
|
||||||
if (error)
|
|
||||||
return (error == -ENXIO ? -ENOENT: error);
|
|
||||||
|
|
||||||
/* Convert next data offset back to a quota id */
|
|
||||||
*id = XFS_B_TO_FSB(mp, offset) * mp->m_quotainfo->qi_dqperchunk;
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -762,7 +767,6 @@ xfs_qm_dqget(
|
|||||||
struct xfs_quotainfo *qi = mp->m_quotainfo;
|
struct xfs_quotainfo *qi = mp->m_quotainfo;
|
||||||
struct radix_tree_root *tree = xfs_dquot_tree(qi, type);
|
struct radix_tree_root *tree = xfs_dquot_tree(qi, type);
|
||||||
struct xfs_dquot *dqp;
|
struct xfs_dquot *dqp;
|
||||||
loff_t eof = 0;
|
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
ASSERT(XFS_IS_QUOTA_RUNNING(mp));
|
ASSERT(XFS_IS_QUOTA_RUNNING(mp));
|
||||||
@ -790,21 +794,6 @@ xfs_qm_dqget(
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Get the end of the quota file if we need it */
|
|
||||||
if (flags & XFS_QMOPT_DQNEXT) {
|
|
||||||
struct xfs_inode *quotip;
|
|
||||||
xfs_fileoff_t last;
|
|
||||||
uint lock_mode;
|
|
||||||
|
|
||||||
quotip = xfs_quota_inode(mp, type);
|
|
||||||
lock_mode = xfs_ilock_data_map_shared(quotip);
|
|
||||||
error = xfs_bmap_last_offset(quotip, &last, XFS_DATA_FORK);
|
|
||||||
xfs_iunlock(quotip, lock_mode);
|
|
||||||
if (error)
|
|
||||||
return error;
|
|
||||||
eof = XFS_FSB_TO_B(mp, last);
|
|
||||||
}
|
|
||||||
|
|
||||||
restart:
|
restart:
|
||||||
mutex_lock(&qi->qi_tree_lock);
|
mutex_lock(&qi->qi_tree_lock);
|
||||||
dqp = radix_tree_lookup(tree, id);
|
dqp = radix_tree_lookup(tree, id);
|
||||||
@ -823,7 +812,7 @@ restart:
|
|||||||
if (XFS_IS_DQUOT_UNINITIALIZED(dqp)) {
|
if (XFS_IS_DQUOT_UNINITIALIZED(dqp)) {
|
||||||
xfs_dqunlock(dqp);
|
xfs_dqunlock(dqp);
|
||||||
mutex_unlock(&qi->qi_tree_lock);
|
mutex_unlock(&qi->qi_tree_lock);
|
||||||
error = xfs_dq_get_next_id(mp, type, &id, eof);
|
error = xfs_dq_get_next_id(mp, type, &id);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
goto restart;
|
goto restart;
|
||||||
@ -858,7 +847,7 @@ restart:
|
|||||||
|
|
||||||
/* If we are asked to find next active id, keep looking */
|
/* If we are asked to find next active id, keep looking */
|
||||||
if (error == -ENOENT && (flags & XFS_QMOPT_DQNEXT)) {
|
if (error == -ENOENT && (flags & XFS_QMOPT_DQNEXT)) {
|
||||||
error = xfs_dq_get_next_id(mp, type, &id, eof);
|
error = xfs_dq_get_next_id(mp, type, &id);
|
||||||
if (!error)
|
if (!error)
|
||||||
goto restart;
|
goto restart;
|
||||||
}
|
}
|
||||||
@ -917,7 +906,7 @@ restart:
|
|||||||
if (flags & XFS_QMOPT_DQNEXT) {
|
if (flags & XFS_QMOPT_DQNEXT) {
|
||||||
if (XFS_IS_DQUOT_UNINITIALIZED(dqp)) {
|
if (XFS_IS_DQUOT_UNINITIALIZED(dqp)) {
|
||||||
xfs_qm_dqput(dqp);
|
xfs_qm_dqput(dqp);
|
||||||
error = xfs_dq_get_next_id(mp, type, &id, eof);
|
error = xfs_dq_get_next_id(mp, type, &id);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
goto restart;
|
goto restart;
|
||||||
|
@ -22,103 +22,280 @@
|
|||||||
#include "xfs_trans_resv.h"
|
#include "xfs_trans_resv.h"
|
||||||
#include "xfs_mount.h"
|
#include "xfs_mount.h"
|
||||||
#include "xfs_error.h"
|
#include "xfs_error.h"
|
||||||
|
#include "xfs_sysfs.h"
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
|
||||||
int xfs_etest[XFS_NUM_INJECT_ERROR];
|
static unsigned int xfs_errortag_random_default[] = {
|
||||||
int64_t xfs_etest_fsid[XFS_NUM_INJECT_ERROR];
|
XFS_RANDOM_DEFAULT,
|
||||||
char * xfs_etest_fsname[XFS_NUM_INJECT_ERROR];
|
XFS_RANDOM_IFLUSH_1,
|
||||||
int xfs_error_test_active;
|
XFS_RANDOM_IFLUSH_2,
|
||||||
|
XFS_RANDOM_IFLUSH_3,
|
||||||
|
XFS_RANDOM_IFLUSH_4,
|
||||||
|
XFS_RANDOM_IFLUSH_5,
|
||||||
|
XFS_RANDOM_IFLUSH_6,
|
||||||
|
XFS_RANDOM_DA_READ_BUF,
|
||||||
|
XFS_RANDOM_BTREE_CHECK_LBLOCK,
|
||||||
|
XFS_RANDOM_BTREE_CHECK_SBLOCK,
|
||||||
|
XFS_RANDOM_ALLOC_READ_AGF,
|
||||||
|
XFS_RANDOM_IALLOC_READ_AGI,
|
||||||
|
XFS_RANDOM_ITOBP_INOTOBP,
|
||||||
|
XFS_RANDOM_IUNLINK,
|
||||||
|
XFS_RANDOM_IUNLINK_REMOVE,
|
||||||
|
XFS_RANDOM_DIR_INO_VALIDATE,
|
||||||
|
XFS_RANDOM_BULKSTAT_READ_CHUNK,
|
||||||
|
XFS_RANDOM_IODONE_IOERR,
|
||||||
|
XFS_RANDOM_STRATREAD_IOERR,
|
||||||
|
XFS_RANDOM_STRATCMPL_IOERR,
|
||||||
|
XFS_RANDOM_DIOWRITE_IOERR,
|
||||||
|
XFS_RANDOM_BMAPIFORMAT,
|
||||||
|
XFS_RANDOM_FREE_EXTENT,
|
||||||
|
XFS_RANDOM_RMAP_FINISH_ONE,
|
||||||
|
XFS_RANDOM_REFCOUNT_CONTINUE_UPDATE,
|
||||||
|
XFS_RANDOM_REFCOUNT_FINISH_ONE,
|
||||||
|
XFS_RANDOM_BMAP_FINISH_ONE,
|
||||||
|
XFS_RANDOM_AG_RESV_CRITICAL,
|
||||||
|
XFS_RANDOM_DROP_WRITES,
|
||||||
|
XFS_RANDOM_LOG_BAD_CRC,
|
||||||
|
};
|
||||||
|
|
||||||
int
|
struct xfs_errortag_attr {
|
||||||
xfs_error_test(int error_tag, int *fsidp, char *expression,
|
struct attribute attr;
|
||||||
int line, char *file, unsigned long randfactor)
|
unsigned int tag;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline struct xfs_errortag_attr *
|
||||||
|
to_attr(struct attribute *attr)
|
||||||
{
|
{
|
||||||
int i;
|
return container_of(attr, struct xfs_errortag_attr, attr);
|
||||||
int64_t fsid;
|
}
|
||||||
|
|
||||||
if (prandom_u32() % randfactor)
|
static inline struct xfs_mount *
|
||||||
return 0;
|
to_mp(struct kobject *kobject)
|
||||||
|
{
|
||||||
|
struct xfs_kobj *kobj = to_kobj(kobject);
|
||||||
|
|
||||||
memcpy(&fsid, fsidp, sizeof(xfs_fsid_t));
|
return container_of(kobj, struct xfs_mount, m_errortag_kobj);
|
||||||
|
}
|
||||||
|
|
||||||
for (i = 0; i < XFS_NUM_INJECT_ERROR; i++) {
|
STATIC ssize_t
|
||||||
if (xfs_etest[i] == error_tag && xfs_etest_fsid[i] == fsid) {
|
xfs_errortag_attr_store(
|
||||||
xfs_warn(NULL,
|
struct kobject *kobject,
|
||||||
"Injecting error (%s) at file %s, line %d, on filesystem \"%s\"",
|
struct attribute *attr,
|
||||||
expression, file, line, xfs_etest_fsname[i]);
|
const char *buf,
|
||||||
return 1;
|
size_t count)
|
||||||
}
|
{
|
||||||
|
struct xfs_mount *mp = to_mp(kobject);
|
||||||
|
struct xfs_errortag_attr *xfs_attr = to_attr(attr);
|
||||||
|
int ret;
|
||||||
|
unsigned int val;
|
||||||
|
|
||||||
|
if (strcmp(buf, "default") == 0) {
|
||||||
|
val = xfs_errortag_random_default[xfs_attr->tag];
|
||||||
|
} else {
|
||||||
|
ret = kstrtouint(buf, 0, &val);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ret = xfs_errortag_set(mp, xfs_attr->tag, val);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
STATIC ssize_t
|
||||||
|
xfs_errortag_attr_show(
|
||||||
|
struct kobject *kobject,
|
||||||
|
struct attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct xfs_mount *mp = to_mp(kobject);
|
||||||
|
struct xfs_errortag_attr *xfs_attr = to_attr(attr);
|
||||||
|
|
||||||
|
return snprintf(buf, PAGE_SIZE, "%u\n",
|
||||||
|
xfs_errortag_get(mp, xfs_attr->tag));
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct sysfs_ops xfs_errortag_sysfs_ops = {
|
||||||
|
.show = xfs_errortag_attr_show,
|
||||||
|
.store = xfs_errortag_attr_store,
|
||||||
|
};
|
||||||
|
|
||||||
|
#define XFS_ERRORTAG_ATTR_RW(_name, _tag) \
|
||||||
|
static struct xfs_errortag_attr xfs_errortag_attr_##_name = { \
|
||||||
|
.attr = {.name = __stringify(_name), \
|
||||||
|
.mode = VERIFY_OCTAL_PERMISSIONS(S_IWUSR | S_IRUGO) }, \
|
||||||
|
.tag = (_tag), \
|
||||||
|
}
|
||||||
|
|
||||||
|
#define XFS_ERRORTAG_ATTR_LIST(_name) &xfs_errortag_attr_##_name.attr
|
||||||
|
|
||||||
|
XFS_ERRORTAG_ATTR_RW(noerror, XFS_ERRTAG_NOERROR);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iflush1, XFS_ERRTAG_IFLUSH_1);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iflush2, XFS_ERRTAG_IFLUSH_2);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iflush3, XFS_ERRTAG_IFLUSH_3);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iflush4, XFS_ERRTAG_IFLUSH_4);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iflush5, XFS_ERRTAG_IFLUSH_5);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iflush6, XFS_ERRTAG_IFLUSH_6);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(dareadbuf, XFS_ERRTAG_DA_READ_BUF);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(btree_chk_lblk, XFS_ERRTAG_BTREE_CHECK_LBLOCK);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(btree_chk_sblk, XFS_ERRTAG_BTREE_CHECK_SBLOCK);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(readagf, XFS_ERRTAG_ALLOC_READ_AGF);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(readagi, XFS_ERRTAG_IALLOC_READ_AGI);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(itobp, XFS_ERRTAG_ITOBP_INOTOBP);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iunlink, XFS_ERRTAG_IUNLINK);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(iunlinkrm, XFS_ERRTAG_IUNLINK_REMOVE);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(dirinovalid, XFS_ERRTAG_DIR_INO_VALIDATE);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(bulkstat, XFS_ERRTAG_BULKSTAT_READ_CHUNK);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(logiodone, XFS_ERRTAG_IODONE_IOERR);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(stratread, XFS_ERRTAG_STRATREAD_IOERR);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(stratcmpl, XFS_ERRTAG_STRATCMPL_IOERR);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(diowrite, XFS_ERRTAG_DIOWRITE_IOERR);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(bmapifmt, XFS_ERRTAG_BMAPIFORMAT);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(free_extent, XFS_ERRTAG_FREE_EXTENT);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(rmap_finish_one, XFS_ERRTAG_RMAP_FINISH_ONE);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(refcount_continue_update, XFS_ERRTAG_REFCOUNT_CONTINUE_UPDATE);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(refcount_finish_one, XFS_ERRTAG_REFCOUNT_FINISH_ONE);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(bmap_finish_one, XFS_ERRTAG_BMAP_FINISH_ONE);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(ag_resv_critical, XFS_ERRTAG_AG_RESV_CRITICAL);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(drop_writes, XFS_ERRTAG_DROP_WRITES);
|
||||||
|
XFS_ERRORTAG_ATTR_RW(log_bad_crc, XFS_ERRTAG_LOG_BAD_CRC);
|
||||||
|
|
||||||
|
static struct attribute *xfs_errortag_attrs[] = {
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(noerror),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iflush1),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iflush2),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iflush3),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iflush4),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iflush5),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iflush6),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(dareadbuf),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(btree_chk_lblk),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(btree_chk_sblk),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(readagf),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(readagi),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(itobp),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iunlink),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(iunlinkrm),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(dirinovalid),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(bulkstat),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(logiodone),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(stratread),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(stratcmpl),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(diowrite),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(bmapifmt),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(free_extent),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(rmap_finish_one),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(refcount_continue_update),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(refcount_finish_one),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(bmap_finish_one),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(ag_resv_critical),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(drop_writes),
|
||||||
|
XFS_ERRORTAG_ATTR_LIST(log_bad_crc),
|
||||||
|
NULL,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct kobj_type xfs_errortag_ktype = {
|
||||||
|
.release = xfs_sysfs_release,
|
||||||
|
.sysfs_ops = &xfs_errortag_sysfs_ops,
|
||||||
|
.default_attrs = xfs_errortag_attrs,
|
||||||
|
};
|
||||||
|
|
||||||
|
int
|
||||||
|
xfs_errortag_init(
|
||||||
|
struct xfs_mount *mp)
|
||||||
|
{
|
||||||
|
mp->m_errortag = kmem_zalloc(sizeof(unsigned int) * XFS_ERRTAG_MAX,
|
||||||
|
KM_SLEEP | KM_MAYFAIL);
|
||||||
|
if (!mp->m_errortag)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
return xfs_sysfs_init(&mp->m_errortag_kobj, &xfs_errortag_ktype,
|
||||||
|
&mp->m_kobj, "errortag");
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
xfs_errortag_del(
|
||||||
|
struct xfs_mount *mp)
|
||||||
|
{
|
||||||
|
xfs_sysfs_del(&mp->m_errortag_kobj);
|
||||||
|
kmem_free(mp->m_errortag);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
xfs_errortag_test(
|
||||||
|
struct xfs_mount *mp,
|
||||||
|
const char *expression,
|
||||||
|
const char *file,
|
||||||
|
int line,
|
||||||
|
unsigned int error_tag)
|
||||||
|
{
|
||||||
|
unsigned int randfactor;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* To be able to use error injection anywhere, we need to ensure error
|
||||||
|
* injection mechanism is already initialized.
|
||||||
|
*
|
||||||
|
* Code paths like I/O completion can be called before the
|
||||||
|
* initialization is complete, but be able to inject errors in such
|
||||||
|
* places is still useful.
|
||||||
|
*/
|
||||||
|
if (!mp->m_errortag)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ASSERT(error_tag < XFS_ERRTAG_MAX);
|
||||||
|
randfactor = mp->m_errortag[error_tag];
|
||||||
|
if (!randfactor || prandom_u32() % randfactor)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
xfs_warn_ratelimited(mp,
|
||||||
|
"Injecting error (%s) at file %s, line %d, on filesystem \"%s\"",
|
||||||
|
expression, file, line, mp->m_fsname);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xfs_errortag_get(
|
||||||
|
struct xfs_mount *mp,
|
||||||
|
unsigned int error_tag)
|
||||||
|
{
|
||||||
|
if (error_tag >= XFS_ERRTAG_MAX)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return mp->m_errortag[error_tag];
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xfs_errortag_set(
|
||||||
|
struct xfs_mount *mp,
|
||||||
|
unsigned int error_tag,
|
||||||
|
unsigned int tag_value)
|
||||||
|
{
|
||||||
|
if (error_tag >= XFS_ERRTAG_MAX)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
mp->m_errortag[error_tag] = tag_value;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
xfs_errortag_add(unsigned int error_tag, xfs_mount_t *mp)
|
xfs_errortag_add(
|
||||||
|
struct xfs_mount *mp,
|
||||||
|
unsigned int error_tag)
|
||||||
{
|
{
|
||||||
int i;
|
|
||||||
int len;
|
|
||||||
int64_t fsid;
|
|
||||||
|
|
||||||
if (error_tag >= XFS_ERRTAG_MAX)
|
if (error_tag >= XFS_ERRTAG_MAX)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
memcpy(&fsid, mp->m_fixedfsid, sizeof(xfs_fsid_t));
|
return xfs_errortag_set(mp, error_tag,
|
||||||
|
xfs_errortag_random_default[error_tag]);
|
||||||
for (i = 0; i < XFS_NUM_INJECT_ERROR; i++) {
|
|
||||||
if (xfs_etest_fsid[i] == fsid && xfs_etest[i] == error_tag) {
|
|
||||||
xfs_warn(mp, "error tag #%d on", error_tag);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; i < XFS_NUM_INJECT_ERROR; i++) {
|
|
||||||
if (xfs_etest[i] == 0) {
|
|
||||||
xfs_warn(mp, "Turned on XFS error tag #%d",
|
|
||||||
error_tag);
|
|
||||||
xfs_etest[i] = error_tag;
|
|
||||||
xfs_etest_fsid[i] = fsid;
|
|
||||||
len = strlen(mp->m_fsname);
|
|
||||||
xfs_etest_fsname[i] = kmem_alloc(len + 1, KM_SLEEP);
|
|
||||||
strcpy(xfs_etest_fsname[i], mp->m_fsname);
|
|
||||||
xfs_error_test_active++;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
xfs_warn(mp, "error tag overflow, too many turned on");
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
xfs_errortag_clearall(xfs_mount_t *mp, int loud)
|
xfs_errortag_clearall(
|
||||||
|
struct xfs_mount *mp)
|
||||||
{
|
{
|
||||||
int64_t fsid;
|
memset(mp->m_errortag, 0, sizeof(unsigned int) * XFS_ERRTAG_MAX);
|
||||||
int cleared = 0;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
memcpy(&fsid, mp->m_fixedfsid, sizeof(xfs_fsid_t));
|
|
||||||
|
|
||||||
|
|
||||||
for (i = 0; i < XFS_NUM_INJECT_ERROR; i++) {
|
|
||||||
if ((fsid == 0LL || xfs_etest_fsid[i] == fsid) &&
|
|
||||||
xfs_etest[i] != 0) {
|
|
||||||
cleared = 1;
|
|
||||||
xfs_warn(mp, "Clearing XFS error tag #%d",
|
|
||||||
xfs_etest[i]);
|
|
||||||
xfs_etest[i] = 0;
|
|
||||||
xfs_etest_fsid[i] = 0LL;
|
|
||||||
kmem_free(xfs_etest_fsname[i]);
|
|
||||||
xfs_etest_fsname[i] = NULL;
|
|
||||||
xfs_error_test_active--;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (loud || cleared)
|
|
||||||
xfs_warn(mp, "Cleared all XFS error tags for filesystem");
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif /* DEBUG */
|
#endif /* DEBUG */
|
||||||
|
@ -96,7 +96,17 @@ extern void xfs_verifier_error(struct xfs_buf *bp);
|
|||||||
#define XFS_ERRTAG_REFCOUNT_FINISH_ONE 25
|
#define XFS_ERRTAG_REFCOUNT_FINISH_ONE 25
|
||||||
#define XFS_ERRTAG_BMAP_FINISH_ONE 26
|
#define XFS_ERRTAG_BMAP_FINISH_ONE 26
|
||||||
#define XFS_ERRTAG_AG_RESV_CRITICAL 27
|
#define XFS_ERRTAG_AG_RESV_CRITICAL 27
|
||||||
#define XFS_ERRTAG_MAX 28
|
/*
|
||||||
|
* DEBUG mode instrumentation to test and/or trigger delayed allocation
|
||||||
|
* block killing in the event of failed writes. When enabled, all
|
||||||
|
* buffered writes are silenty dropped and handled as if they failed.
|
||||||
|
* All delalloc blocks in the range of the write (including pre-existing
|
||||||
|
* delalloc blocks!) are tossed as part of the write failure error
|
||||||
|
* handling sequence.
|
||||||
|
*/
|
||||||
|
#define XFS_ERRTAG_DROP_WRITES 28
|
||||||
|
#define XFS_ERRTAG_LOG_BAD_CRC 29
|
||||||
|
#define XFS_ERRTAG_MAX 30
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Random factors for above tags, 1 means always, 2 means 1/2 time, etc.
|
* Random factors for above tags, 1 means always, 2 means 1/2 time, etc.
|
||||||
@ -129,23 +139,29 @@ extern void xfs_verifier_error(struct xfs_buf *bp);
|
|||||||
#define XFS_RANDOM_REFCOUNT_FINISH_ONE 1
|
#define XFS_RANDOM_REFCOUNT_FINISH_ONE 1
|
||||||
#define XFS_RANDOM_BMAP_FINISH_ONE 1
|
#define XFS_RANDOM_BMAP_FINISH_ONE 1
|
||||||
#define XFS_RANDOM_AG_RESV_CRITICAL 4
|
#define XFS_RANDOM_AG_RESV_CRITICAL 4
|
||||||
|
#define XFS_RANDOM_DROP_WRITES 1
|
||||||
|
#define XFS_RANDOM_LOG_BAD_CRC 1
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
extern int xfs_error_test_active;
|
extern int xfs_errortag_init(struct xfs_mount *mp);
|
||||||
extern int xfs_error_test(int, int *, char *, int, char *, unsigned long);
|
extern void xfs_errortag_del(struct xfs_mount *mp);
|
||||||
|
extern bool xfs_errortag_test(struct xfs_mount *mp, const char *expression,
|
||||||
|
const char *file, int line, unsigned int error_tag);
|
||||||
|
#define XFS_TEST_ERROR(expr, mp, tag) \
|
||||||
|
((expr) || xfs_errortag_test((mp), #expr, __FILE__, __LINE__, (tag)))
|
||||||
|
|
||||||
#define XFS_NUM_INJECT_ERROR 10
|
extern int xfs_errortag_get(struct xfs_mount *mp, unsigned int error_tag);
|
||||||
#define XFS_TEST_ERROR(expr, mp, tag, rf) \
|
extern int xfs_errortag_set(struct xfs_mount *mp, unsigned int error_tag,
|
||||||
((expr) || (xfs_error_test_active && \
|
unsigned int tag_value);
|
||||||
xfs_error_test((tag), (mp)->m_fixedfsid, "expr", __LINE__, __FILE__, \
|
extern int xfs_errortag_add(struct xfs_mount *mp, unsigned int error_tag);
|
||||||
(rf))))
|
extern int xfs_errortag_clearall(struct xfs_mount *mp);
|
||||||
|
|
||||||
extern int xfs_errortag_add(unsigned int error_tag, struct xfs_mount *mp);
|
|
||||||
extern int xfs_errortag_clearall(struct xfs_mount *mp, int loud);
|
|
||||||
#else
|
#else
|
||||||
#define XFS_TEST_ERROR(expr, mp, tag, rf) (expr)
|
#define xfs_errortag_init(mp) (0)
|
||||||
#define xfs_errortag_add(tag, mp) (ENOSYS)
|
#define xfs_errortag_del(mp)
|
||||||
#define xfs_errortag_clearall(mp, loud) (ENOSYS)
|
#define XFS_TEST_ERROR(expr, mp, tag) (expr)
|
||||||
|
#define xfs_errortag_set(mp, tag, val) (ENOSYS)
|
||||||
|
#define xfs_errortag_add(mp, tag) (ENOSYS)
|
||||||
|
#define xfs_errortag_clearall(mp) (ENOSYS)
|
||||||
#endif /* DEBUG */
|
#endif /* DEBUG */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -679,6 +679,7 @@ write_retry:
|
|||||||
xfs_iunlock(ip, iolock);
|
xfs_iunlock(ip, iolock);
|
||||||
eofb.eof_flags = XFS_EOF_FLAGS_SYNC;
|
eofb.eof_flags = XFS_EOF_FLAGS_SYNC;
|
||||||
xfs_icache_free_eofblocks(ip->i_mount, &eofb);
|
xfs_icache_free_eofblocks(ip->i_mount, &eofb);
|
||||||
|
xfs_icache_free_cowblocks(ip->i_mount, &eofb);
|
||||||
goto write_retry;
|
goto write_retry;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -970,362 +971,7 @@ xfs_file_readdir(
|
|||||||
*/
|
*/
|
||||||
bufsize = (size_t)min_t(loff_t, 32768, ip->i_d.di_size);
|
bufsize = (size_t)min_t(loff_t, 32768, ip->i_d.di_size);
|
||||||
|
|
||||||
return xfs_readdir(ip, ctx, bufsize);
|
return xfs_readdir(NULL, ip, ctx, bufsize);
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This type is designed to indicate the type of offset we would like
|
|
||||||
* to search from page cache for xfs_seek_hole_data().
|
|
||||||
*/
|
|
||||||
enum {
|
|
||||||
HOLE_OFF = 0,
|
|
||||||
DATA_OFF,
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Lookup the desired type of offset from the given page.
|
|
||||||
*
|
|
||||||
* On success, return true and the offset argument will point to the
|
|
||||||
* start of the region that was found. Otherwise this function will
|
|
||||||
* return false and keep the offset argument unchanged.
|
|
||||||
*/
|
|
||||||
STATIC bool
|
|
||||||
xfs_lookup_buffer_offset(
|
|
||||||
struct page *page,
|
|
||||||
loff_t *offset,
|
|
||||||
unsigned int type)
|
|
||||||
{
|
|
||||||
loff_t lastoff = page_offset(page);
|
|
||||||
bool found = false;
|
|
||||||
struct buffer_head *bh, *head;
|
|
||||||
|
|
||||||
bh = head = page_buffers(page);
|
|
||||||
do {
|
|
||||||
/*
|
|
||||||
* Unwritten extents that have data in the page
|
|
||||||
* cache covering them can be identified by the
|
|
||||||
* BH_Unwritten state flag. Pages with multiple
|
|
||||||
* buffers might have a mix of holes, data and
|
|
||||||
* unwritten extents - any buffer with valid
|
|
||||||
* data in it should have BH_Uptodate flag set
|
|
||||||
* on it.
|
|
||||||
*/
|
|
||||||
if (buffer_unwritten(bh) ||
|
|
||||||
buffer_uptodate(bh)) {
|
|
||||||
if (type == DATA_OFF)
|
|
||||||
found = true;
|
|
||||||
} else {
|
|
||||||
if (type == HOLE_OFF)
|
|
||||||
found = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (found) {
|
|
||||||
*offset = lastoff;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
lastoff += bh->b_size;
|
|
||||||
} while ((bh = bh->b_this_page) != head);
|
|
||||||
|
|
||||||
return found;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This routine is called to find out and return a data or hole offset
|
|
||||||
* from the page cache for unwritten extents according to the desired
|
|
||||||
* type for xfs_seek_hole_data().
|
|
||||||
*
|
|
||||||
* The argument offset is used to tell where we start to search from the
|
|
||||||
* page cache. Map is used to figure out the end points of the range to
|
|
||||||
* lookup pages.
|
|
||||||
*
|
|
||||||
* Return true if the desired type of offset was found, and the argument
|
|
||||||
* offset is filled with that address. Otherwise, return false and keep
|
|
||||||
* offset unchanged.
|
|
||||||
*/
|
|
||||||
STATIC bool
|
|
||||||
xfs_find_get_desired_pgoff(
|
|
||||||
struct inode *inode,
|
|
||||||
struct xfs_bmbt_irec *map,
|
|
||||||
unsigned int type,
|
|
||||||
loff_t *offset)
|
|
||||||
{
|
|
||||||
struct xfs_inode *ip = XFS_I(inode);
|
|
||||||
struct xfs_mount *mp = ip->i_mount;
|
|
||||||
struct pagevec pvec;
|
|
||||||
pgoff_t index;
|
|
||||||
pgoff_t end;
|
|
||||||
loff_t endoff;
|
|
||||||
loff_t startoff = *offset;
|
|
||||||
loff_t lastoff = startoff;
|
|
||||||
bool found = false;
|
|
||||||
|
|
||||||
pagevec_init(&pvec, 0);
|
|
||||||
|
|
||||||
index = startoff >> PAGE_SHIFT;
|
|
||||||
endoff = XFS_FSB_TO_B(mp, map->br_startoff + map->br_blockcount);
|
|
||||||
end = (endoff - 1) >> PAGE_SHIFT;
|
|
||||||
do {
|
|
||||||
int want;
|
|
||||||
unsigned nr_pages;
|
|
||||||
unsigned int i;
|
|
||||||
|
|
||||||
want = min_t(pgoff_t, end - index, PAGEVEC_SIZE - 1) + 1;
|
|
||||||
nr_pages = pagevec_lookup(&pvec, inode->i_mapping, index,
|
|
||||||
want);
|
|
||||||
if (nr_pages == 0)
|
|
||||||
break;
|
|
||||||
|
|
||||||
for (i = 0; i < nr_pages; i++) {
|
|
||||||
struct page *page = pvec.pages[i];
|
|
||||||
loff_t b_offset;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* At this point, the page may be truncated or
|
|
||||||
* invalidated (changing page->mapping to NULL),
|
|
||||||
* or even swizzled back from swapper_space to tmpfs
|
|
||||||
* file mapping. However, page->index will not change
|
|
||||||
* because we have a reference on the page.
|
|
||||||
*
|
|
||||||
* If current page offset is beyond where we've ended,
|
|
||||||
* we've found a hole.
|
|
||||||
*/
|
|
||||||
if (type == HOLE_OFF && lastoff < endoff &&
|
|
||||||
lastoff < page_offset(pvec.pages[i])) {
|
|
||||||
found = true;
|
|
||||||
*offset = lastoff;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
/* Searching done if the page index is out of range. */
|
|
||||||
if (page->index > end)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
lock_page(page);
|
|
||||||
/*
|
|
||||||
* Page truncated or invalidated(page->mapping == NULL).
|
|
||||||
* We can freely skip it and proceed to check the next
|
|
||||||
* page.
|
|
||||||
*/
|
|
||||||
if (unlikely(page->mapping != inode->i_mapping)) {
|
|
||||||
unlock_page(page);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!page_has_buffers(page)) {
|
|
||||||
unlock_page(page);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
found = xfs_lookup_buffer_offset(page, &b_offset, type);
|
|
||||||
if (found) {
|
|
||||||
/*
|
|
||||||
* The found offset may be less than the start
|
|
||||||
* point to search if this is the first time to
|
|
||||||
* come here.
|
|
||||||
*/
|
|
||||||
*offset = max_t(loff_t, startoff, b_offset);
|
|
||||||
unlock_page(page);
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* We either searching data but nothing was found, or
|
|
||||||
* searching hole but found a data buffer. In either
|
|
||||||
* case, probably the next page contains the desired
|
|
||||||
* things, update the last offset to it so.
|
|
||||||
*/
|
|
||||||
lastoff = page_offset(page) + PAGE_SIZE;
|
|
||||||
unlock_page(page);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* The number of returned pages less than our desired, search
|
|
||||||
* done.
|
|
||||||
*/
|
|
||||||
if (nr_pages < want)
|
|
||||||
break;
|
|
||||||
|
|
||||||
index = pvec.pages[i - 1]->index + 1;
|
|
||||||
pagevec_release(&pvec);
|
|
||||||
} while (index <= end);
|
|
||||||
|
|
||||||
/* No page at lastoff and we are not done - we found a hole. */
|
|
||||||
if (type == HOLE_OFF && lastoff < endoff) {
|
|
||||||
*offset = lastoff;
|
|
||||||
found = true;
|
|
||||||
}
|
|
||||||
out:
|
|
||||||
pagevec_release(&pvec);
|
|
||||||
return found;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* caller must lock inode with xfs_ilock_data_map_shared,
|
|
||||||
* can we craft an appropriate ASSERT?
|
|
||||||
*
|
|
||||||
* end is because the VFS-level lseek interface is defined such that any
|
|
||||||
* offset past i_size shall return -ENXIO, but we use this for quota code
|
|
||||||
* which does not maintain i_size, and we want to SEEK_DATA past i_size.
|
|
||||||
*/
|
|
||||||
loff_t
|
|
||||||
__xfs_seek_hole_data(
|
|
||||||
struct inode *inode,
|
|
||||||
loff_t start,
|
|
||||||
loff_t end,
|
|
||||||
int whence)
|
|
||||||
{
|
|
||||||
struct xfs_inode *ip = XFS_I(inode);
|
|
||||||
struct xfs_mount *mp = ip->i_mount;
|
|
||||||
loff_t uninitialized_var(offset);
|
|
||||||
xfs_fileoff_t fsbno;
|
|
||||||
xfs_filblks_t lastbno;
|
|
||||||
int error;
|
|
||||||
|
|
||||||
if (start >= end) {
|
|
||||||
error = -ENXIO;
|
|
||||||
goto out_error;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Try to read extents from the first block indicated
|
|
||||||
* by fsbno to the end block of the file.
|
|
||||||
*/
|
|
||||||
fsbno = XFS_B_TO_FSBT(mp, start);
|
|
||||||
lastbno = XFS_B_TO_FSB(mp, end);
|
|
||||||
|
|
||||||
for (;;) {
|
|
||||||
struct xfs_bmbt_irec map[2];
|
|
||||||
int nmap = 2;
|
|
||||||
unsigned int i;
|
|
||||||
|
|
||||||
error = xfs_bmapi_read(ip, fsbno, lastbno - fsbno, map, &nmap,
|
|
||||||
XFS_BMAPI_ENTIRE);
|
|
||||||
if (error)
|
|
||||||
goto out_error;
|
|
||||||
|
|
||||||
/* No extents at given offset, must be beyond EOF */
|
|
||||||
if (nmap == 0) {
|
|
||||||
error = -ENXIO;
|
|
||||||
goto out_error;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; i < nmap; i++) {
|
|
||||||
offset = max_t(loff_t, start,
|
|
||||||
XFS_FSB_TO_B(mp, map[i].br_startoff));
|
|
||||||
|
|
||||||
/* Landed in the hole we wanted? */
|
|
||||||
if (whence == SEEK_HOLE &&
|
|
||||||
map[i].br_startblock == HOLESTARTBLOCK)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
/* Landed in the data extent we wanted? */
|
|
||||||
if (whence == SEEK_DATA &&
|
|
||||||
(map[i].br_startblock == DELAYSTARTBLOCK ||
|
|
||||||
(map[i].br_state == XFS_EXT_NORM &&
|
|
||||||
!isnullstartblock(map[i].br_startblock))))
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Landed in an unwritten extent, try to search
|
|
||||||
* for hole or data from page cache.
|
|
||||||
*/
|
|
||||||
if (map[i].br_state == XFS_EXT_UNWRITTEN) {
|
|
||||||
if (xfs_find_get_desired_pgoff(inode, &map[i],
|
|
||||||
whence == SEEK_HOLE ? HOLE_OFF : DATA_OFF,
|
|
||||||
&offset))
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* We only received one extent out of the two requested. This
|
|
||||||
* means we've hit EOF and didn't find what we are looking for.
|
|
||||||
*/
|
|
||||||
if (nmap == 1) {
|
|
||||||
/*
|
|
||||||
* If we were looking for a hole, set offset to
|
|
||||||
* the end of the file (i.e., there is an implicit
|
|
||||||
* hole at the end of any file).
|
|
||||||
*/
|
|
||||||
if (whence == SEEK_HOLE) {
|
|
||||||
offset = end;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
/*
|
|
||||||
* If we were looking for data, it's nowhere to be found
|
|
||||||
*/
|
|
||||||
ASSERT(whence == SEEK_DATA);
|
|
||||||
error = -ENXIO;
|
|
||||||
goto out_error;
|
|
||||||
}
|
|
||||||
|
|
||||||
ASSERT(i > 1);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Nothing was found, proceed to the next round of search
|
|
||||||
* if the next reading offset is not at or beyond EOF.
|
|
||||||
*/
|
|
||||||
fsbno = map[i - 1].br_startoff + map[i - 1].br_blockcount;
|
|
||||||
start = XFS_FSB_TO_B(mp, fsbno);
|
|
||||||
if (start >= end) {
|
|
||||||
if (whence == SEEK_HOLE) {
|
|
||||||
offset = end;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
ASSERT(whence == SEEK_DATA);
|
|
||||||
error = -ENXIO;
|
|
||||||
goto out_error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
out:
|
|
||||||
/*
|
|
||||||
* If at this point we have found the hole we wanted, the returned
|
|
||||||
* offset may be bigger than the file size as it may be aligned to
|
|
||||||
* page boundary for unwritten extents. We need to deal with this
|
|
||||||
* situation in particular.
|
|
||||||
*/
|
|
||||||
if (whence == SEEK_HOLE)
|
|
||||||
offset = min_t(loff_t, offset, end);
|
|
||||||
|
|
||||||
return offset;
|
|
||||||
|
|
||||||
out_error:
|
|
||||||
return error;
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC loff_t
|
|
||||||
xfs_seek_hole_data(
|
|
||||||
struct file *file,
|
|
||||||
loff_t start,
|
|
||||||
int whence)
|
|
||||||
{
|
|
||||||
struct inode *inode = file->f_mapping->host;
|
|
||||||
struct xfs_inode *ip = XFS_I(inode);
|
|
||||||
struct xfs_mount *mp = ip->i_mount;
|
|
||||||
uint lock;
|
|
||||||
loff_t offset, end;
|
|
||||||
int error = 0;
|
|
||||||
|
|
||||||
if (XFS_FORCED_SHUTDOWN(mp))
|
|
||||||
return -EIO;
|
|
||||||
|
|
||||||
lock = xfs_ilock_data_map_shared(ip);
|
|
||||||
|
|
||||||
end = i_size_read(inode);
|
|
||||||
offset = __xfs_seek_hole_data(inode, start, end, whence);
|
|
||||||
if (offset < 0) {
|
|
||||||
error = offset;
|
|
||||||
goto out_unlock;
|
|
||||||
}
|
|
||||||
|
|
||||||
offset = vfs_setpos(file, offset, inode->i_sb->s_maxbytes);
|
|
||||||
|
|
||||||
out_unlock:
|
|
||||||
xfs_iunlock(ip, lock);
|
|
||||||
|
|
||||||
if (error)
|
|
||||||
return error;
|
|
||||||
return offset;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC loff_t
|
STATIC loff_t
|
||||||
@ -1334,17 +980,25 @@ xfs_file_llseek(
|
|||||||
loff_t offset,
|
loff_t offset,
|
||||||
int whence)
|
int whence)
|
||||||
{
|
{
|
||||||
|
struct inode *inode = file->f_mapping->host;
|
||||||
|
|
||||||
|
if (XFS_FORCED_SHUTDOWN(XFS_I(inode)->i_mount))
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
switch (whence) {
|
switch (whence) {
|
||||||
case SEEK_END:
|
default:
|
||||||
case SEEK_CUR:
|
|
||||||
case SEEK_SET:
|
|
||||||
return generic_file_llseek(file, offset, whence);
|
return generic_file_llseek(file, offset, whence);
|
||||||
case SEEK_HOLE:
|
case SEEK_HOLE:
|
||||||
|
offset = iomap_seek_hole(inode, offset, &xfs_iomap_ops);
|
||||||
|
break;
|
||||||
case SEEK_DATA:
|
case SEEK_DATA:
|
||||||
return xfs_seek_hole_data(file, offset, whence);
|
offset = iomap_seek_data(inode, offset, &xfs_iomap_ops);
|
||||||
default:
|
break;
|
||||||
return -EINVAL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (offset < 0)
|
||||||
|
return offset;
|
||||||
|
return vfs_setpos(file, offset, inode->i_sb->s_maxbytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -602,7 +602,7 @@ xfs_growfs_data_private(
|
|||||||
if (nagimax)
|
if (nagimax)
|
||||||
mp->m_maxagi = nagimax;
|
mp->m_maxagi = nagimax;
|
||||||
if (mp->m_sb.sb_imax_pct) {
|
if (mp->m_sb.sb_imax_pct) {
|
||||||
__uint64_t icount = mp->m_sb.sb_dblocks * mp->m_sb.sb_imax_pct;
|
uint64_t icount = mp->m_sb.sb_dblocks * mp->m_sb.sb_imax_pct;
|
||||||
do_div(icount, 100);
|
do_div(icount, 100);
|
||||||
mp->m_maxicount = icount << mp->m_sb.sb_inopblog;
|
mp->m_maxicount = icount << mp->m_sb.sb_inopblog;
|
||||||
} else
|
} else
|
||||||
@ -793,17 +793,17 @@ xfs_fs_counts(
|
|||||||
int
|
int
|
||||||
xfs_reserve_blocks(
|
xfs_reserve_blocks(
|
||||||
xfs_mount_t *mp,
|
xfs_mount_t *mp,
|
||||||
__uint64_t *inval,
|
uint64_t *inval,
|
||||||
xfs_fsop_resblks_t *outval)
|
xfs_fsop_resblks_t *outval)
|
||||||
{
|
{
|
||||||
__int64_t lcounter, delta;
|
int64_t lcounter, delta;
|
||||||
__int64_t fdblks_delta = 0;
|
int64_t fdblks_delta = 0;
|
||||||
__uint64_t request;
|
uint64_t request;
|
||||||
__int64_t free;
|
int64_t free;
|
||||||
int error = 0;
|
int error = 0;
|
||||||
|
|
||||||
/* If inval is null, report current values and return */
|
/* If inval is null, report current values and return */
|
||||||
if (inval == (__uint64_t *)NULL) {
|
if (inval == (uint64_t *)NULL) {
|
||||||
if (!outval)
|
if (!outval)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
outval->resblks = mp->m_resblks;
|
outval->resblks = mp->m_resblks;
|
||||||
@ -904,7 +904,7 @@ out:
|
|||||||
int
|
int
|
||||||
xfs_fs_goingdown(
|
xfs_fs_goingdown(
|
||||||
xfs_mount_t *mp,
|
xfs_mount_t *mp,
|
||||||
__uint32_t inflags)
|
uint32_t inflags)
|
||||||
{
|
{
|
||||||
switch (inflags) {
|
switch (inflags) {
|
||||||
case XFS_FSOP_GOING_FLAGS_DEFAULT: {
|
case XFS_FSOP_GOING_FLAGS_DEFAULT: {
|
||||||
|
@ -22,9 +22,9 @@ extern int xfs_fs_geometry(xfs_mount_t *mp, xfs_fsop_geom_t *geo, int nversion);
|
|||||||
extern int xfs_growfs_data(xfs_mount_t *mp, xfs_growfs_data_t *in);
|
extern int xfs_growfs_data(xfs_mount_t *mp, xfs_growfs_data_t *in);
|
||||||
extern int xfs_growfs_log(xfs_mount_t *mp, xfs_growfs_log_t *in);
|
extern int xfs_growfs_log(xfs_mount_t *mp, xfs_growfs_log_t *in);
|
||||||
extern int xfs_fs_counts(xfs_mount_t *mp, xfs_fsop_counts_t *cnt);
|
extern int xfs_fs_counts(xfs_mount_t *mp, xfs_fsop_counts_t *cnt);
|
||||||
extern int xfs_reserve_blocks(xfs_mount_t *mp, __uint64_t *inval,
|
extern int xfs_reserve_blocks(xfs_mount_t *mp, uint64_t *inval,
|
||||||
xfs_fsop_resblks_t *outval);
|
xfs_fsop_resblks_t *outval);
|
||||||
extern int xfs_fs_goingdown(xfs_mount_t *mp, __uint32_t inflags);
|
extern int xfs_fs_goingdown(xfs_mount_t *mp, uint32_t inflags);
|
||||||
|
|
||||||
extern int xfs_fs_reserve_ag_blocks(struct xfs_mount *mp);
|
extern int xfs_fs_reserve_ag_blocks(struct xfs_mount *mp);
|
||||||
extern int xfs_fs_unreserve_ag_blocks(struct xfs_mount *mp);
|
extern int xfs_fs_unreserve_ag_blocks(struct xfs_mount *mp);
|
||||||
|
@ -47,4 +47,9 @@ xfs_param_t xfs_params = {
|
|||||||
|
|
||||||
struct xfs_globals xfs_globals = {
|
struct xfs_globals xfs_globals = {
|
||||||
.log_recovery_delay = 0, /* no delay by default */
|
.log_recovery_delay = 0, /* no delay by default */
|
||||||
|
#ifdef XFS_ASSERT_FATAL
|
||||||
|
.bug_on_assert = true, /* assert failures BUG() */
|
||||||
|
#else
|
||||||
|
.bug_on_assert = false, /* assert failures WARN() */
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
@ -368,6 +368,11 @@ xfs_iget_cache_hit(
|
|||||||
if (ip->i_flags & XFS_IRECLAIMABLE) {
|
if (ip->i_flags & XFS_IRECLAIMABLE) {
|
||||||
trace_xfs_iget_reclaim(ip);
|
trace_xfs_iget_reclaim(ip);
|
||||||
|
|
||||||
|
if (flags & XFS_IGET_INCORE) {
|
||||||
|
error = -EAGAIN;
|
||||||
|
goto out_error;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We need to set XFS_IRECLAIM to prevent xfs_reclaim_inode
|
* We need to set XFS_IRECLAIM to prevent xfs_reclaim_inode
|
||||||
* from stomping over us while we recycle the inode. We can't
|
* from stomping over us while we recycle the inode. We can't
|
||||||
@ -432,7 +437,8 @@ xfs_iget_cache_hit(
|
|||||||
if (lock_flags != 0)
|
if (lock_flags != 0)
|
||||||
xfs_ilock(ip, lock_flags);
|
xfs_ilock(ip, lock_flags);
|
||||||
|
|
||||||
xfs_iflags_clear(ip, XFS_ISTALE | XFS_IDONTCACHE);
|
if (!(flags & XFS_IGET_INCORE))
|
||||||
|
xfs_iflags_clear(ip, XFS_ISTALE | XFS_IDONTCACHE);
|
||||||
XFS_STATS_INC(mp, xs_ig_found);
|
XFS_STATS_INC(mp, xs_ig_found);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -603,6 +609,10 @@ again:
|
|||||||
goto out_error_or_again;
|
goto out_error_or_again;
|
||||||
} else {
|
} else {
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
|
if (flags & XFS_IGET_INCORE) {
|
||||||
|
error = -ENOENT;
|
||||||
|
goto out_error_or_again;
|
||||||
|
}
|
||||||
XFS_STATS_INC(mp, xs_ig_missed);
|
XFS_STATS_INC(mp, xs_ig_missed);
|
||||||
|
|
||||||
error = xfs_iget_cache_miss(mp, pag, tp, ino, &ip,
|
error = xfs_iget_cache_miss(mp, pag, tp, ino, &ip,
|
||||||
@ -623,7 +633,7 @@ again:
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
out_error_or_again:
|
out_error_or_again:
|
||||||
if (error == -EAGAIN) {
|
if (!(flags & XFS_IGET_INCORE) && error == -EAGAIN) {
|
||||||
delay(1);
|
delay(1);
|
||||||
goto again;
|
goto again;
|
||||||
}
|
}
|
||||||
@ -631,6 +641,44 @@ out_error_or_again:
|
|||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* "Is this a cached inode that's also allocated?"
|
||||||
|
*
|
||||||
|
* Look up an inode by number in the given file system. If the inode is
|
||||||
|
* in cache and isn't in purgatory, return 1 if the inode is allocated
|
||||||
|
* and 0 if it is not. For all other cases (not in cache, being torn
|
||||||
|
* down, etc.), return a negative error code.
|
||||||
|
*
|
||||||
|
* The caller has to prevent inode allocation and freeing activity,
|
||||||
|
* presumably by locking the AGI buffer. This is to ensure that an
|
||||||
|
* inode cannot transition from allocated to freed until the caller is
|
||||||
|
* ready to allow that. If the inode is in an intermediate state (new,
|
||||||
|
* reclaimable, or being reclaimed), -EAGAIN will be returned; if the
|
||||||
|
* inode is not in the cache, -ENOENT will be returned. The caller must
|
||||||
|
* deal with these scenarios appropriately.
|
||||||
|
*
|
||||||
|
* This is a specialized use case for the online scrubber; if you're
|
||||||
|
* reading this, you probably want xfs_iget.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
xfs_icache_inode_is_allocated(
|
||||||
|
struct xfs_mount *mp,
|
||||||
|
struct xfs_trans *tp,
|
||||||
|
xfs_ino_t ino,
|
||||||
|
bool *inuse)
|
||||||
|
{
|
||||||
|
struct xfs_inode *ip;
|
||||||
|
int error;
|
||||||
|
|
||||||
|
error = xfs_iget(mp, tp, ino, XFS_IGET_INCORE, 0, &ip);
|
||||||
|
if (error)
|
||||||
|
return error;
|
||||||
|
|
||||||
|
*inuse = !!(VFS_I(ip)->i_mode);
|
||||||
|
IRELE(ip);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The inode lookup is done in batches to keep the amount of lock traffic and
|
* The inode lookup is done in batches to keep the amount of lock traffic and
|
||||||
* radix tree lookups to a minimum. The batch size is a trade off between
|
* radix tree lookups to a minimum. The batch size is a trade off between
|
||||||
|
@ -47,6 +47,7 @@ struct xfs_eofblocks {
|
|||||||
#define XFS_IGET_CREATE 0x1
|
#define XFS_IGET_CREATE 0x1
|
||||||
#define XFS_IGET_UNTRUSTED 0x2
|
#define XFS_IGET_UNTRUSTED 0x2
|
||||||
#define XFS_IGET_DONTCACHE 0x4
|
#define XFS_IGET_DONTCACHE 0x4
|
||||||
|
#define XFS_IGET_INCORE 0x8 /* don't read from disk or reinit */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* flags for AG inode iterator
|
* flags for AG inode iterator
|
||||||
@ -126,4 +127,7 @@ xfs_fs_eofblocks_from_user(
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int xfs_icache_inode_is_allocated(struct xfs_mount *mp, struct xfs_trans *tp,
|
||||||
|
xfs_ino_t ino, bool *inuse);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -632,7 +632,7 @@ __xfs_iflock(
|
|||||||
|
|
||||||
STATIC uint
|
STATIC uint
|
||||||
_xfs_dic2xflags(
|
_xfs_dic2xflags(
|
||||||
__uint16_t di_flags,
|
uint16_t di_flags,
|
||||||
uint64_t di_flags2,
|
uint64_t di_flags2,
|
||||||
bool has_attr)
|
bool has_attr)
|
||||||
{
|
{
|
||||||
@ -855,8 +855,8 @@ xfs_ialloc(
|
|||||||
inode->i_version = 1;
|
inode->i_version = 1;
|
||||||
ip->i_d.di_flags2 = 0;
|
ip->i_d.di_flags2 = 0;
|
||||||
ip->i_d.di_cowextsize = 0;
|
ip->i_d.di_cowextsize = 0;
|
||||||
ip->i_d.di_crtime.t_sec = (__int32_t)tv.tv_sec;
|
ip->i_d.di_crtime.t_sec = (int32_t)tv.tv_sec;
|
||||||
ip->i_d.di_crtime.t_nsec = (__int32_t)tv.tv_nsec;
|
ip->i_d.di_crtime.t_nsec = (int32_t)tv.tv_nsec;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -3489,7 +3489,7 @@ xfs_iflush_int(
|
|||||||
dip = xfs_buf_offset(bp, ip->i_imap.im_boffset);
|
dip = xfs_buf_offset(bp, ip->i_imap.im_boffset);
|
||||||
|
|
||||||
if (XFS_TEST_ERROR(dip->di_magic != cpu_to_be16(XFS_DINODE_MAGIC),
|
if (XFS_TEST_ERROR(dip->di_magic != cpu_to_be16(XFS_DINODE_MAGIC),
|
||||||
mp, XFS_ERRTAG_IFLUSH_1, XFS_RANDOM_IFLUSH_1)) {
|
mp, XFS_ERRTAG_IFLUSH_1)) {
|
||||||
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
||||||
"%s: Bad inode %Lu magic number 0x%x, ptr 0x%p",
|
"%s: Bad inode %Lu magic number 0x%x, ptr 0x%p",
|
||||||
__func__, ip->i_ino, be16_to_cpu(dip->di_magic), dip);
|
__func__, ip->i_ino, be16_to_cpu(dip->di_magic), dip);
|
||||||
@ -3499,7 +3499,7 @@ xfs_iflush_int(
|
|||||||
if (XFS_TEST_ERROR(
|
if (XFS_TEST_ERROR(
|
||||||
(ip->i_d.di_format != XFS_DINODE_FMT_EXTENTS) &&
|
(ip->i_d.di_format != XFS_DINODE_FMT_EXTENTS) &&
|
||||||
(ip->i_d.di_format != XFS_DINODE_FMT_BTREE),
|
(ip->i_d.di_format != XFS_DINODE_FMT_BTREE),
|
||||||
mp, XFS_ERRTAG_IFLUSH_3, XFS_RANDOM_IFLUSH_3)) {
|
mp, XFS_ERRTAG_IFLUSH_3)) {
|
||||||
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
||||||
"%s: Bad regular inode %Lu, ptr 0x%p",
|
"%s: Bad regular inode %Lu, ptr 0x%p",
|
||||||
__func__, ip->i_ino, ip);
|
__func__, ip->i_ino, ip);
|
||||||
@ -3510,7 +3510,7 @@ xfs_iflush_int(
|
|||||||
(ip->i_d.di_format != XFS_DINODE_FMT_EXTENTS) &&
|
(ip->i_d.di_format != XFS_DINODE_FMT_EXTENTS) &&
|
||||||
(ip->i_d.di_format != XFS_DINODE_FMT_BTREE) &&
|
(ip->i_d.di_format != XFS_DINODE_FMT_BTREE) &&
|
||||||
(ip->i_d.di_format != XFS_DINODE_FMT_LOCAL),
|
(ip->i_d.di_format != XFS_DINODE_FMT_LOCAL),
|
||||||
mp, XFS_ERRTAG_IFLUSH_4, XFS_RANDOM_IFLUSH_4)) {
|
mp, XFS_ERRTAG_IFLUSH_4)) {
|
||||||
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
||||||
"%s: Bad directory inode %Lu, ptr 0x%p",
|
"%s: Bad directory inode %Lu, ptr 0x%p",
|
||||||
__func__, ip->i_ino, ip);
|
__func__, ip->i_ino, ip);
|
||||||
@ -3518,8 +3518,7 @@ xfs_iflush_int(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (XFS_TEST_ERROR(ip->i_d.di_nextents + ip->i_d.di_anextents >
|
if (XFS_TEST_ERROR(ip->i_d.di_nextents + ip->i_d.di_anextents >
|
||||||
ip->i_d.di_nblocks, mp, XFS_ERRTAG_IFLUSH_5,
|
ip->i_d.di_nblocks, mp, XFS_ERRTAG_IFLUSH_5)) {
|
||||||
XFS_RANDOM_IFLUSH_5)) {
|
|
||||||
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
||||||
"%s: detected corrupt incore inode %Lu, "
|
"%s: detected corrupt incore inode %Lu, "
|
||||||
"total extents = %d, nblocks = %Ld, ptr 0x%p",
|
"total extents = %d, nblocks = %Ld, ptr 0x%p",
|
||||||
@ -3529,7 +3528,7 @@ xfs_iflush_int(
|
|||||||
goto corrupt_out;
|
goto corrupt_out;
|
||||||
}
|
}
|
||||||
if (XFS_TEST_ERROR(ip->i_d.di_forkoff > mp->m_sb.sb_inodesize,
|
if (XFS_TEST_ERROR(ip->i_d.di_forkoff > mp->m_sb.sb_inodesize,
|
||||||
mp, XFS_ERRTAG_IFLUSH_6, XFS_RANDOM_IFLUSH_6)) {
|
mp, XFS_ERRTAG_IFLUSH_6)) {
|
||||||
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
xfs_alert_tag(mp, XFS_PTAG_IFLUSH,
|
||||||
"%s: bad inode %Lu, forkoff 0x%x, ptr 0x%p",
|
"%s: bad inode %Lu, forkoff 0x%x, ptr 0x%p",
|
||||||
__func__, ip->i_ino, ip->i_d.di_forkoff, ip);
|
__func__, ip->i_ino, ip->i_d.di_forkoff, ip);
|
||||||
|
@ -192,8 +192,8 @@ static inline void
|
|||||||
xfs_set_projid(struct xfs_inode *ip,
|
xfs_set_projid(struct xfs_inode *ip,
|
||||||
prid_t projid)
|
prid_t projid)
|
||||||
{
|
{
|
||||||
ip->i_d.di_projid_hi = (__uint16_t) (projid >> 16);
|
ip->i_d.di_projid_hi = (uint16_t) (projid >> 16);
|
||||||
ip->i_d.di_projid_lo = (__uint16_t) (projid & 0xffff);
|
ip->i_d.di_projid_lo = (uint16_t) (projid & 0xffff);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline prid_t
|
static inline prid_t
|
||||||
@ -445,9 +445,6 @@ int xfs_zero_eof(struct xfs_inode *ip, xfs_off_t offset,
|
|||||||
xfs_fsize_t isize, bool *did_zeroing);
|
xfs_fsize_t isize, bool *did_zeroing);
|
||||||
int xfs_zero_range(struct xfs_inode *ip, xfs_off_t pos, xfs_off_t count,
|
int xfs_zero_range(struct xfs_inode *ip, xfs_off_t pos, xfs_off_t count,
|
||||||
bool *did_zero);
|
bool *did_zero);
|
||||||
loff_t __xfs_seek_hole_data(struct inode *inode, loff_t start,
|
|
||||||
loff_t eof, int whence);
|
|
||||||
|
|
||||||
|
|
||||||
/* from xfs_iops.c */
|
/* from xfs_iops.c */
|
||||||
extern void xfs_setup_inode(struct xfs_inode *ip);
|
extern void xfs_setup_inode(struct xfs_inode *ip);
|
||||||
|
@ -120,8 +120,7 @@ xfs_find_handle(
|
|||||||
handle.ha_fid.fid_pad = 0;
|
handle.ha_fid.fid_pad = 0;
|
||||||
handle.ha_fid.fid_gen = inode->i_generation;
|
handle.ha_fid.fid_gen = inode->i_generation;
|
||||||
handle.ha_fid.fid_ino = ip->i_ino;
|
handle.ha_fid.fid_ino = ip->i_ino;
|
||||||
|
hsize = sizeof(xfs_handle_t);
|
||||||
hsize = XFS_HSIZE(handle);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
error = -EFAULT;
|
error = -EFAULT;
|
||||||
@ -444,8 +443,8 @@ xfs_attrmulti_attr_get(
|
|||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
unsigned char *name,
|
unsigned char *name,
|
||||||
unsigned char __user *ubuf,
|
unsigned char __user *ubuf,
|
||||||
__uint32_t *len,
|
uint32_t *len,
|
||||||
__uint32_t flags)
|
uint32_t flags)
|
||||||
{
|
{
|
||||||
unsigned char *kbuf;
|
unsigned char *kbuf;
|
||||||
int error = -EFAULT;
|
int error = -EFAULT;
|
||||||
@ -473,8 +472,8 @@ xfs_attrmulti_attr_set(
|
|||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
unsigned char *name,
|
unsigned char *name,
|
||||||
const unsigned char __user *ubuf,
|
const unsigned char __user *ubuf,
|
||||||
__uint32_t len,
|
uint32_t len,
|
||||||
__uint32_t flags)
|
uint32_t flags)
|
||||||
{
|
{
|
||||||
unsigned char *kbuf;
|
unsigned char *kbuf;
|
||||||
int error;
|
int error;
|
||||||
@ -499,7 +498,7 @@ int
|
|||||||
xfs_attrmulti_attr_remove(
|
xfs_attrmulti_attr_remove(
|
||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
unsigned char *name,
|
unsigned char *name,
|
||||||
__uint32_t flags)
|
uint32_t flags)
|
||||||
{
|
{
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
@ -877,7 +876,7 @@ xfs_merge_ioc_xflags(
|
|||||||
|
|
||||||
STATIC unsigned int
|
STATIC unsigned int
|
||||||
xfs_di2lxflags(
|
xfs_di2lxflags(
|
||||||
__uint16_t di_flags)
|
uint16_t di_flags)
|
||||||
{
|
{
|
||||||
unsigned int flags = 0;
|
unsigned int flags = 0;
|
||||||
|
|
||||||
@ -1288,7 +1287,7 @@ xfs_ioctl_setattr_check_projid(
|
|||||||
struct fsxattr *fa)
|
struct fsxattr *fa)
|
||||||
{
|
{
|
||||||
/* Disallow 32bit project ids if projid32bit feature is not enabled. */
|
/* Disallow 32bit project ids if projid32bit feature is not enabled. */
|
||||||
if (fa->fsx_projid > (__uint16_t)-1 &&
|
if (fa->fsx_projid > (uint16_t)-1 &&
|
||||||
!xfs_sb_version_hasprojid32bit(&ip->i_mount->m_sb))
|
!xfs_sb_version_hasprojid32bit(&ip->i_mount->m_sb))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
@ -1932,7 +1931,7 @@ xfs_file_ioctl(
|
|||||||
|
|
||||||
case XFS_IOC_SET_RESBLKS: {
|
case XFS_IOC_SET_RESBLKS: {
|
||||||
xfs_fsop_resblks_t inout;
|
xfs_fsop_resblks_t inout;
|
||||||
__uint64_t in;
|
uint64_t in;
|
||||||
|
|
||||||
if (!capable(CAP_SYS_ADMIN))
|
if (!capable(CAP_SYS_ADMIN))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
@ -2018,12 +2017,12 @@ xfs_file_ioctl(
|
|||||||
}
|
}
|
||||||
|
|
||||||
case XFS_IOC_GOINGDOWN: {
|
case XFS_IOC_GOINGDOWN: {
|
||||||
__uint32_t in;
|
uint32_t in;
|
||||||
|
|
||||||
if (!capable(CAP_SYS_ADMIN))
|
if (!capable(CAP_SYS_ADMIN))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
|
|
||||||
if (get_user(in, (__uint32_t __user *)arg))
|
if (get_user(in, (uint32_t __user *)arg))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
return xfs_fs_goingdown(mp, in);
|
return xfs_fs_goingdown(mp, in);
|
||||||
@ -2038,14 +2037,14 @@ xfs_file_ioctl(
|
|||||||
if (copy_from_user(&in, arg, sizeof(in)))
|
if (copy_from_user(&in, arg, sizeof(in)))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
return xfs_errortag_add(in.errtag, mp);
|
return xfs_errortag_add(mp, in.errtag);
|
||||||
}
|
}
|
||||||
|
|
||||||
case XFS_IOC_ERROR_CLEARALL:
|
case XFS_IOC_ERROR_CLEARALL:
|
||||||
if (!capable(CAP_SYS_ADMIN))
|
if (!capable(CAP_SYS_ADMIN))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
|
|
||||||
return xfs_errortag_clearall(mp, 1);
|
return xfs_errortag_clearall(mp);
|
||||||
|
|
||||||
case XFS_IOC_FREE_EOFBLOCKS: {
|
case XFS_IOC_FREE_EOFBLOCKS: {
|
||||||
struct xfs_fs_eofblocks eofb;
|
struct xfs_fs_eofblocks eofb;
|
||||||
|
@ -48,22 +48,22 @@ xfs_attrmulti_attr_get(
|
|||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
unsigned char *name,
|
unsigned char *name,
|
||||||
unsigned char __user *ubuf,
|
unsigned char __user *ubuf,
|
||||||
__uint32_t *len,
|
uint32_t *len,
|
||||||
__uint32_t flags);
|
uint32_t flags);
|
||||||
|
|
||||||
extern int
|
extern int
|
||||||
xfs_attrmulti_attr_set(
|
xfs_attrmulti_attr_set(
|
||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
unsigned char *name,
|
unsigned char *name,
|
||||||
const unsigned char __user *ubuf,
|
const unsigned char __user *ubuf,
|
||||||
__uint32_t len,
|
uint32_t len,
|
||||||
__uint32_t flags);
|
uint32_t flags);
|
||||||
|
|
||||||
extern int
|
extern int
|
||||||
xfs_attrmulti_attr_remove(
|
xfs_attrmulti_attr_remove(
|
||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
unsigned char *name,
|
unsigned char *name,
|
||||||
__uint32_t flags);
|
uint32_t flags);
|
||||||
|
|
||||||
extern struct dentry *
|
extern struct dentry *
|
||||||
xfs_handle_to_dentry(
|
xfs_handle_to_dentry(
|
||||||
|
@ -112,9 +112,9 @@ typedef struct compat_xfs_fsop_handlereq {
|
|||||||
|
|
||||||
/* The bstat field in the swapext struct needs translation */
|
/* The bstat field in the swapext struct needs translation */
|
||||||
typedef struct compat_xfs_swapext {
|
typedef struct compat_xfs_swapext {
|
||||||
__int64_t sx_version; /* version */
|
int64_t sx_version; /* version */
|
||||||
__int64_t sx_fdtarget; /* fd of target file */
|
int64_t sx_fdtarget; /* fd of target file */
|
||||||
__int64_t sx_fdtmp; /* fd of tmp file */
|
int64_t sx_fdtmp; /* fd of tmp file */
|
||||||
xfs_off_t sx_offset; /* offset into file */
|
xfs_off_t sx_offset; /* offset into file */
|
||||||
xfs_off_t sx_length; /* leng from offset */
|
xfs_off_t sx_length; /* leng from offset */
|
||||||
char sx_pad[16]; /* pad space, unused */
|
char sx_pad[16]; /* pad space, unused */
|
||||||
|
@ -543,7 +543,7 @@ xfs_file_iomap_begin_delay(
|
|||||||
if (unlikely(XFS_TEST_ERROR(
|
if (unlikely(XFS_TEST_ERROR(
|
||||||
(XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_EXTENTS &&
|
(XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_EXTENTS &&
|
||||||
XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_BTREE),
|
XFS_IFORK_FORMAT(ip, XFS_DATA_FORK) != XFS_DINODE_FMT_BTREE),
|
||||||
mp, XFS_ERRTAG_BMAPIFORMAT, XFS_RANDOM_BMAPIFORMAT))) {
|
mp, XFS_ERRTAG_BMAPIFORMAT))) {
|
||||||
XFS_ERROR_REPORT(__func__, XFS_ERRLEVEL_LOW, mp);
|
XFS_ERROR_REPORT(__func__, XFS_ERRLEVEL_LOW, mp);
|
||||||
error = -EFSCORRUPTED;
|
error = -EFSCORRUPTED;
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
@ -1119,7 +1119,7 @@ xfs_file_iomap_end_delalloc(
|
|||||||
* Behave as if the write failed if drop writes is enabled. Set the NEW
|
* Behave as if the write failed if drop writes is enabled. Set the NEW
|
||||||
* flag to force delalloc cleanup.
|
* flag to force delalloc cleanup.
|
||||||
*/
|
*/
|
||||||
if (xfs_mp_drop_writes(mp)) {
|
if (XFS_TEST_ERROR(false, mp, XFS_ERRTAG_DROP_WRITES)) {
|
||||||
iomap->flags |= IOMAP_F_NEW;
|
iomap->flags |= IOMAP_F_NEW;
|
||||||
written = 0;
|
written = 0;
|
||||||
}
|
}
|
||||||
|
@ -190,12 +190,12 @@ xfs_generic_create(
|
|||||||
|
|
||||||
#ifdef CONFIG_XFS_POSIX_ACL
|
#ifdef CONFIG_XFS_POSIX_ACL
|
||||||
if (default_acl) {
|
if (default_acl) {
|
||||||
error = xfs_set_acl(inode, default_acl, ACL_TYPE_DEFAULT);
|
error = __xfs_set_acl(inode, default_acl, ACL_TYPE_DEFAULT);
|
||||||
if (error)
|
if (error)
|
||||||
goto out_cleanup_inode;
|
goto out_cleanup_inode;
|
||||||
}
|
}
|
||||||
if (acl) {
|
if (acl) {
|
||||||
error = xfs_set_acl(inode, acl, ACL_TYPE_ACCESS);
|
error = __xfs_set_acl(inode, acl, ACL_TYPE_ACCESS);
|
||||||
if (error)
|
if (error)
|
||||||
goto out_cleanup_inode;
|
goto out_cleanup_inode;
|
||||||
}
|
}
|
||||||
@ -460,7 +460,7 @@ xfs_vn_get_link(
|
|||||||
if (!dentry)
|
if (!dentry)
|
||||||
return ERR_PTR(-ECHILD);
|
return ERR_PTR(-ECHILD);
|
||||||
|
|
||||||
link = kmalloc(MAXPATHLEN+1, GFP_KERNEL);
|
link = kmalloc(XFS_SYMLINK_MAXLEN+1, GFP_KERNEL);
|
||||||
if (!link)
|
if (!link)
|
||||||
goto out_err;
|
goto out_err;
|
||||||
|
|
||||||
|
@ -31,7 +31,7 @@
|
|||||||
#include "xfs_trace.h"
|
#include "xfs_trace.h"
|
||||||
#include "xfs_icache.h"
|
#include "xfs_icache.h"
|
||||||
|
|
||||||
STATIC int
|
int
|
||||||
xfs_internal_inum(
|
xfs_internal_inum(
|
||||||
xfs_mount_t *mp,
|
xfs_mount_t *mp,
|
||||||
xfs_ino_t ino)
|
xfs_ino_t ino)
|
||||||
|
@ -96,4 +96,6 @@ xfs_inumbers(
|
|||||||
void __user *buffer, /* buffer with inode info */
|
void __user *buffer, /* buffer with inode info */
|
||||||
inumbers_fmt_pf formatter);
|
inumbers_fmt_pf formatter);
|
||||||
|
|
||||||
|
int xfs_internal_inum(struct xfs_mount *mp, xfs_ino_t ino);
|
||||||
|
|
||||||
#endif /* __XFS_ITABLE_H__ */
|
#endif /* __XFS_ITABLE_H__ */
|
||||||
|
@ -24,14 +24,6 @@
|
|||||||
/*
|
/*
|
||||||
* Kernel specific type declarations for XFS
|
* Kernel specific type declarations for XFS
|
||||||
*/
|
*/
|
||||||
typedef signed char __int8_t;
|
|
||||||
typedef unsigned char __uint8_t;
|
|
||||||
typedef signed short int __int16_t;
|
|
||||||
typedef unsigned short int __uint16_t;
|
|
||||||
typedef signed int __int32_t;
|
|
||||||
typedef unsigned int __uint32_t;
|
|
||||||
typedef signed long long int __int64_t;
|
|
||||||
typedef unsigned long long int __uint64_t;
|
|
||||||
|
|
||||||
typedef __s64 xfs_off_t; /* <file offset> type */
|
typedef __s64 xfs_off_t; /* <file offset> type */
|
||||||
typedef unsigned long long xfs_ino_t; /* <inode> type */
|
typedef unsigned long long xfs_ino_t; /* <inode> type */
|
||||||
@ -151,7 +143,6 @@ typedef __u32 xfs_nlink_t;
|
|||||||
#define __return_address __builtin_return_address(0)
|
#define __return_address __builtin_return_address(0)
|
||||||
|
|
||||||
#define XFS_PROJID_DEFAULT 0
|
#define XFS_PROJID_DEFAULT 0
|
||||||
#define MAXPATHLEN 1024
|
|
||||||
|
|
||||||
#define MIN(a,b) (min(a,b))
|
#define MIN(a,b) (min(a,b))
|
||||||
#define MAX(a,b) (max(a,b))
|
#define MAX(a,b) (max(a,b))
|
||||||
@ -186,22 +177,22 @@ extern struct xstats xfsstats;
|
|||||||
* are converting to the init_user_ns. The uid is later mapped to a particular
|
* are converting to the init_user_ns. The uid is later mapped to a particular
|
||||||
* user namespace value when crossing the kernel/user boundary.
|
* user namespace value when crossing the kernel/user boundary.
|
||||||
*/
|
*/
|
||||||
static inline __uint32_t xfs_kuid_to_uid(kuid_t uid)
|
static inline uint32_t xfs_kuid_to_uid(kuid_t uid)
|
||||||
{
|
{
|
||||||
return from_kuid(&init_user_ns, uid);
|
return from_kuid(&init_user_ns, uid);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline kuid_t xfs_uid_to_kuid(__uint32_t uid)
|
static inline kuid_t xfs_uid_to_kuid(uint32_t uid)
|
||||||
{
|
{
|
||||||
return make_kuid(&init_user_ns, uid);
|
return make_kuid(&init_user_ns, uid);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline __uint32_t xfs_kgid_to_gid(kgid_t gid)
|
static inline uint32_t xfs_kgid_to_gid(kgid_t gid)
|
||||||
{
|
{
|
||||||
return from_kgid(&init_user_ns, gid);
|
return from_kgid(&init_user_ns, gid);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline kgid_t xfs_gid_to_kgid(__uint32_t gid)
|
static inline kgid_t xfs_gid_to_kgid(uint32_t gid)
|
||||||
{
|
{
|
||||||
return make_kgid(&init_user_ns, gid);
|
return make_kgid(&init_user_ns, gid);
|
||||||
}
|
}
|
||||||
@ -231,14 +222,14 @@ static inline __u32 xfs_do_mod(void *a, __u32 b, int n)
|
|||||||
|
|
||||||
#define do_mod(a, b) xfs_do_mod(&(a), (b), sizeof(a))
|
#define do_mod(a, b) xfs_do_mod(&(a), (b), sizeof(a))
|
||||||
|
|
||||||
static inline __uint64_t roundup_64(__uint64_t x, __uint32_t y)
|
static inline uint64_t roundup_64(uint64_t x, uint32_t y)
|
||||||
{
|
{
|
||||||
x += y - 1;
|
x += y - 1;
|
||||||
do_div(x, y);
|
do_div(x, y);
|
||||||
return x * y;
|
return x * y;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline __uint64_t howmany_64(__uint64_t x, __uint32_t y)
|
static inline uint64_t howmany_64(uint64_t x, uint32_t y)
|
||||||
{
|
{
|
||||||
x += y - 1;
|
x += y - 1;
|
||||||
do_div(x, y);
|
do_div(x, y);
|
||||||
|
@ -434,7 +434,7 @@ xfs_log_reserve(
|
|||||||
int unit_bytes,
|
int unit_bytes,
|
||||||
int cnt,
|
int cnt,
|
||||||
struct xlog_ticket **ticp,
|
struct xlog_ticket **ticp,
|
||||||
__uint8_t client,
|
uint8_t client,
|
||||||
bool permanent)
|
bool permanent)
|
||||||
{
|
{
|
||||||
struct xlog *log = mp->m_log;
|
struct xlog *log = mp->m_log;
|
||||||
@ -825,9 +825,9 @@ xfs_log_unmount_write(xfs_mount_t *mp)
|
|||||||
if (!error) {
|
if (!error) {
|
||||||
/* the data section must be 32 bit size aligned */
|
/* the data section must be 32 bit size aligned */
|
||||||
struct {
|
struct {
|
||||||
__uint16_t magic;
|
uint16_t magic;
|
||||||
__uint16_t pad1;
|
uint16_t pad1;
|
||||||
__uint32_t pad2; /* may as well make it 64 bits */
|
uint32_t pad2; /* may as well make it 64 bits */
|
||||||
} magic = {
|
} magic = {
|
||||||
.magic = XLOG_UNMOUNT_TYPE,
|
.magic = XLOG_UNMOUNT_TYPE,
|
||||||
};
|
};
|
||||||
@ -1189,8 +1189,7 @@ xlog_iodone(xfs_buf_t *bp)
|
|||||||
* IOABORT state. The IOABORT state is only set in DEBUG mode to inject
|
* IOABORT state. The IOABORT state is only set in DEBUG mode to inject
|
||||||
* CRC errors into log recovery.
|
* CRC errors into log recovery.
|
||||||
*/
|
*/
|
||||||
if (XFS_TEST_ERROR(bp->b_error, l->l_mp, XFS_ERRTAG_IODONE_IOERR,
|
if (XFS_TEST_ERROR(bp->b_error, l->l_mp, XFS_ERRTAG_IODONE_IOERR) ||
|
||||||
XFS_RANDOM_IODONE_IOERR) ||
|
|
||||||
iclog->ic_state & XLOG_STATE_IOABORT) {
|
iclog->ic_state & XLOG_STATE_IOABORT) {
|
||||||
if (iclog->ic_state & XLOG_STATE_IOABORT)
|
if (iclog->ic_state & XLOG_STATE_IOABORT)
|
||||||
iclog->ic_state &= ~XLOG_STATE_IOABORT;
|
iclog->ic_state &= ~XLOG_STATE_IOABORT;
|
||||||
@ -1665,7 +1664,7 @@ xlog_cksum(
|
|||||||
char *dp,
|
char *dp,
|
||||||
int size)
|
int size)
|
||||||
{
|
{
|
||||||
__uint32_t crc;
|
uint32_t crc;
|
||||||
|
|
||||||
/* first generate the crc for the record header ... */
|
/* first generate the crc for the record header ... */
|
||||||
crc = xfs_start_cksum_update((char *)rhead,
|
crc = xfs_start_cksum_update((char *)rhead,
|
||||||
@ -1828,7 +1827,7 @@ xlog_sync(
|
|||||||
*/
|
*/
|
||||||
dptr = (char *)&iclog->ic_header + count;
|
dptr = (char *)&iclog->ic_header + count;
|
||||||
for (i = 0; i < split; i += BBSIZE) {
|
for (i = 0; i < split; i += BBSIZE) {
|
||||||
__uint32_t cycle = be32_to_cpu(*(__be32 *)dptr);
|
uint32_t cycle = be32_to_cpu(*(__be32 *)dptr);
|
||||||
if (++cycle == XLOG_HEADER_MAGIC_NUM)
|
if (++cycle == XLOG_HEADER_MAGIC_NUM)
|
||||||
cycle++;
|
cycle++;
|
||||||
*(__be32 *)dptr = cpu_to_be32(cycle);
|
*(__be32 *)dptr = cpu_to_be32(cycle);
|
||||||
@ -1842,7 +1841,6 @@ xlog_sync(
|
|||||||
/* calculcate the checksum */
|
/* calculcate the checksum */
|
||||||
iclog->ic_header.h_crc = xlog_cksum(log, &iclog->ic_header,
|
iclog->ic_header.h_crc = xlog_cksum(log, &iclog->ic_header,
|
||||||
iclog->ic_datap, size);
|
iclog->ic_datap, size);
|
||||||
#ifdef DEBUG
|
|
||||||
/*
|
/*
|
||||||
* Intentionally corrupt the log record CRC based on the error injection
|
* Intentionally corrupt the log record CRC based on the error injection
|
||||||
* frequency, if defined. This facilitates testing log recovery in the
|
* frequency, if defined. This facilitates testing log recovery in the
|
||||||
@ -1850,15 +1848,13 @@ xlog_sync(
|
|||||||
* write on I/O completion and shutdown the fs. The subsequent mount
|
* write on I/O completion and shutdown the fs. The subsequent mount
|
||||||
* detects the bad CRC and attempts to recover.
|
* detects the bad CRC and attempts to recover.
|
||||||
*/
|
*/
|
||||||
if (log->l_badcrc_factor &&
|
if (XFS_TEST_ERROR(false, log->l_mp, XFS_ERRTAG_LOG_BAD_CRC)) {
|
||||||
(prandom_u32() % log->l_badcrc_factor == 0)) {
|
|
||||||
iclog->ic_header.h_crc &= cpu_to_le32(0xAAAAAAAA);
|
iclog->ic_header.h_crc &= cpu_to_le32(0xAAAAAAAA);
|
||||||
iclog->ic_state |= XLOG_STATE_IOABORT;
|
iclog->ic_state |= XLOG_STATE_IOABORT;
|
||||||
xfs_warn(log->l_mp,
|
xfs_warn(log->l_mp,
|
||||||
"Intentionally corrupted log record at LSN 0x%llx. Shutdown imminent.",
|
"Intentionally corrupted log record at LSN 0x%llx. Shutdown imminent.",
|
||||||
be64_to_cpu(iclog->ic_header.h_lsn));
|
be64_to_cpu(iclog->ic_header.h_lsn));
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
bp->b_io_length = BTOBB(count);
|
bp->b_io_length = BTOBB(count);
|
||||||
bp->b_fspriv = iclog;
|
bp->b_fspriv = iclog;
|
||||||
@ -2024,7 +2020,7 @@ xlog_print_tic_res(
|
|||||||
};
|
};
|
||||||
#undef REG_TYPE_STR
|
#undef REG_TYPE_STR
|
||||||
|
|
||||||
xfs_warn(mp, "xlog_write: reservation summary:");
|
xfs_warn(mp, "ticket reservation summary:");
|
||||||
xfs_warn(mp, " unit res = %d bytes",
|
xfs_warn(mp, " unit res = %d bytes",
|
||||||
ticket->t_unit_res);
|
ticket->t_unit_res);
|
||||||
xfs_warn(mp, " current res = %d bytes",
|
xfs_warn(mp, " current res = %d bytes",
|
||||||
@ -2045,10 +2041,55 @@ xlog_print_tic_res(
|
|||||||
"bad-rtype" : res_type_str[r_type]),
|
"bad-rtype" : res_type_str[r_type]),
|
||||||
ticket->t_res_arr[i].r_len);
|
ticket->t_res_arr[i].r_len);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
xfs_alert_tag(mp, XFS_PTAG_LOGRES,
|
/*
|
||||||
"xlog_write: reservation ran out. Need to up reservation");
|
* Print a summary of the transaction.
|
||||||
xfs_force_shutdown(mp, SHUTDOWN_LOG_IO_ERROR);
|
*/
|
||||||
|
void
|
||||||
|
xlog_print_trans(
|
||||||
|
struct xfs_trans *tp)
|
||||||
|
{
|
||||||
|
struct xfs_mount *mp = tp->t_mountp;
|
||||||
|
struct xfs_log_item_desc *lidp;
|
||||||
|
|
||||||
|
/* dump core transaction and ticket info */
|
||||||
|
xfs_warn(mp, "transaction summary:");
|
||||||
|
xfs_warn(mp, " flags = 0x%x", tp->t_flags);
|
||||||
|
|
||||||
|
xlog_print_tic_res(mp, tp->t_ticket);
|
||||||
|
|
||||||
|
/* dump each log item */
|
||||||
|
list_for_each_entry(lidp, &tp->t_items, lid_trans) {
|
||||||
|
struct xfs_log_item *lip = lidp->lid_item;
|
||||||
|
struct xfs_log_vec *lv = lip->li_lv;
|
||||||
|
struct xfs_log_iovec *vec;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
xfs_warn(mp, "log item: ");
|
||||||
|
xfs_warn(mp, " type = 0x%x", lip->li_type);
|
||||||
|
xfs_warn(mp, " flags = 0x%x", lip->li_flags);
|
||||||
|
if (!lv)
|
||||||
|
continue;
|
||||||
|
xfs_warn(mp, " niovecs = %d", lv->lv_niovecs);
|
||||||
|
xfs_warn(mp, " size = %d", lv->lv_size);
|
||||||
|
xfs_warn(mp, " bytes = %d", lv->lv_bytes);
|
||||||
|
xfs_warn(mp, " buf len = %d", lv->lv_buf_len);
|
||||||
|
|
||||||
|
/* dump each iovec for the log item */
|
||||||
|
vec = lv->lv_iovecp;
|
||||||
|
for (i = 0; i < lv->lv_niovecs; i++) {
|
||||||
|
int dumplen = min(vec->i_len, 32);
|
||||||
|
|
||||||
|
xfs_warn(mp, " iovec[%d]", i);
|
||||||
|
xfs_warn(mp, " type = 0x%x", vec->i_type);
|
||||||
|
xfs_warn(mp, " len = %d", vec->i_len);
|
||||||
|
xfs_warn(mp, " first %d bytes of iovec[%d]:", dumplen, i);
|
||||||
|
xfs_hex_dump(vec->i_addr, dumplen);
|
||||||
|
|
||||||
|
vec++;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2321,8 +2362,12 @@ xlog_write(
|
|||||||
if (flags & (XLOG_COMMIT_TRANS | XLOG_UNMOUNT_TRANS))
|
if (flags & (XLOG_COMMIT_TRANS | XLOG_UNMOUNT_TRANS))
|
||||||
ticket->t_curr_res -= sizeof(xlog_op_header_t);
|
ticket->t_curr_res -= sizeof(xlog_op_header_t);
|
||||||
|
|
||||||
if (ticket->t_curr_res < 0)
|
if (ticket->t_curr_res < 0) {
|
||||||
|
xfs_alert_tag(log->l_mp, XFS_PTAG_LOGRES,
|
||||||
|
"ctx ticket reservation ran out. Need to up reservation");
|
||||||
xlog_print_tic_res(log->l_mp, ticket);
|
xlog_print_tic_res(log->l_mp, ticket);
|
||||||
|
xfs_force_shutdown(log->l_mp, SHUTDOWN_LOG_IO_ERROR);
|
||||||
|
}
|
||||||
|
|
||||||
index = 0;
|
index = 0;
|
||||||
lv = log_vector;
|
lv = log_vector;
|
||||||
@ -2363,8 +2408,8 @@ xlog_write(
|
|||||||
}
|
}
|
||||||
|
|
||||||
reg = &vecp[index];
|
reg = &vecp[index];
|
||||||
ASSERT(reg->i_len % sizeof(__int32_t) == 0);
|
ASSERT(reg->i_len % sizeof(int32_t) == 0);
|
||||||
ASSERT((unsigned long)ptr % sizeof(__int32_t) == 0);
|
ASSERT((unsigned long)ptr % sizeof(int32_t) == 0);
|
||||||
|
|
||||||
start_rec_copy = xlog_write_start_rec(ptr, ticket);
|
start_rec_copy = xlog_write_start_rec(ptr, ticket);
|
||||||
if (start_rec_copy) {
|
if (start_rec_copy) {
|
||||||
@ -3143,7 +3188,7 @@ xlog_state_switch_iclogs(
|
|||||||
/* Round up to next log-sunit */
|
/* Round up to next log-sunit */
|
||||||
if (xfs_sb_version_haslogv2(&log->l_mp->m_sb) &&
|
if (xfs_sb_version_haslogv2(&log->l_mp->m_sb) &&
|
||||||
log->l_mp->m_sb.sb_logsunit > 1) {
|
log->l_mp->m_sb.sb_logsunit > 1) {
|
||||||
__uint32_t sunit_bb = BTOBB(log->l_mp->m_sb.sb_logsunit);
|
uint32_t sunit_bb = BTOBB(log->l_mp->m_sb.sb_logsunit);
|
||||||
log->l_curr_block = roundup(log->l_curr_block, sunit_bb);
|
log->l_curr_block = roundup(log->l_curr_block, sunit_bb);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3771,7 +3816,7 @@ xlog_verify_iclog(
|
|||||||
xlog_in_core_2_t *xhdr;
|
xlog_in_core_2_t *xhdr;
|
||||||
void *base_ptr, *ptr, *p;
|
void *base_ptr, *ptr, *p;
|
||||||
ptrdiff_t field_offset;
|
ptrdiff_t field_offset;
|
||||||
__uint8_t clientid;
|
uint8_t clientid;
|
||||||
int len, i, j, k, op_len;
|
int len, i, j, k, op_len;
|
||||||
int idx;
|
int idx;
|
||||||
|
|
||||||
|
@ -159,7 +159,7 @@ int xfs_log_reserve(struct xfs_mount *mp,
|
|||||||
int length,
|
int length,
|
||||||
int count,
|
int count,
|
||||||
struct xlog_ticket **ticket,
|
struct xlog_ticket **ticket,
|
||||||
__uint8_t clientid,
|
uint8_t clientid,
|
||||||
bool permanent);
|
bool permanent);
|
||||||
int xfs_log_regrant(struct xfs_mount *mp, struct xlog_ticket *tic);
|
int xfs_log_regrant(struct xfs_mount *mp, struct xlog_ticket *tic);
|
||||||
void xfs_log_unmount(struct xfs_mount *mp);
|
void xfs_log_unmount(struct xfs_mount *mp);
|
||||||
|
@ -410,6 +410,7 @@ xlog_cil_insert_items(
|
|||||||
int len = 0;
|
int len = 0;
|
||||||
int diff_iovecs = 0;
|
int diff_iovecs = 0;
|
||||||
int iclog_space;
|
int iclog_space;
|
||||||
|
int iovhdr_res = 0, split_res = 0, ctx_res = 0;
|
||||||
|
|
||||||
ASSERT(tp);
|
ASSERT(tp);
|
||||||
|
|
||||||
@ -419,12 +420,65 @@ xlog_cil_insert_items(
|
|||||||
*/
|
*/
|
||||||
xlog_cil_insert_format_items(log, tp, &len, &diff_iovecs);
|
xlog_cil_insert_format_items(log, tp, &len, &diff_iovecs);
|
||||||
|
|
||||||
|
spin_lock(&cil->xc_cil_lock);
|
||||||
|
|
||||||
|
/* account for space used by new iovec headers */
|
||||||
|
iovhdr_res = diff_iovecs * sizeof(xlog_op_header_t);
|
||||||
|
len += iovhdr_res;
|
||||||
|
ctx->nvecs += diff_iovecs;
|
||||||
|
|
||||||
|
/* attach the transaction to the CIL if it has any busy extents */
|
||||||
|
if (!list_empty(&tp->t_busy))
|
||||||
|
list_splice_init(&tp->t_busy, &ctx->busy_extents);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Now transfer enough transaction reservation to the context ticket
|
||||||
|
* for the checkpoint. The context ticket is special - the unit
|
||||||
|
* reservation has to grow as well as the current reservation as we
|
||||||
|
* steal from tickets so we can correctly determine the space used
|
||||||
|
* during the transaction commit.
|
||||||
|
*/
|
||||||
|
if (ctx->ticket->t_curr_res == 0) {
|
||||||
|
ctx_res = ctx->ticket->t_unit_res;
|
||||||
|
ctx->ticket->t_curr_res = ctx_res;
|
||||||
|
tp->t_ticket->t_curr_res -= ctx_res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* do we need space for more log record headers? */
|
||||||
|
iclog_space = log->l_iclog_size - log->l_iclog_hsize;
|
||||||
|
if (len > 0 && (ctx->space_used / iclog_space !=
|
||||||
|
(ctx->space_used + len) / iclog_space)) {
|
||||||
|
split_res = (len + iclog_space - 1) / iclog_space;
|
||||||
|
/* need to take into account split region headers, too */
|
||||||
|
split_res *= log->l_iclog_hsize + sizeof(struct xlog_op_header);
|
||||||
|
ctx->ticket->t_unit_res += split_res;
|
||||||
|
ctx->ticket->t_curr_res += split_res;
|
||||||
|
tp->t_ticket->t_curr_res -= split_res;
|
||||||
|
ASSERT(tp->t_ticket->t_curr_res >= len);
|
||||||
|
}
|
||||||
|
tp->t_ticket->t_curr_res -= len;
|
||||||
|
ctx->space_used += len;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we've overrun the reservation, dump the tx details before we move
|
||||||
|
* the log items. Shutdown is imminent...
|
||||||
|
*/
|
||||||
|
if (WARN_ON(tp->t_ticket->t_curr_res < 0)) {
|
||||||
|
xfs_warn(log->l_mp, "Transaction log reservation overrun:");
|
||||||
|
xfs_warn(log->l_mp,
|
||||||
|
" log items: %d bytes (iov hdrs: %d bytes)",
|
||||||
|
len, iovhdr_res);
|
||||||
|
xfs_warn(log->l_mp, " split region headers: %d bytes",
|
||||||
|
split_res);
|
||||||
|
xfs_warn(log->l_mp, " ctx ticket: %d bytes", ctx_res);
|
||||||
|
xlog_print_trans(tp);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Now (re-)position everything modified at the tail of the CIL.
|
* Now (re-)position everything modified at the tail of the CIL.
|
||||||
* We do this here so we only need to take the CIL lock once during
|
* We do this here so we only need to take the CIL lock once during
|
||||||
* the transaction commit.
|
* the transaction commit.
|
||||||
*/
|
*/
|
||||||
spin_lock(&cil->xc_cil_lock);
|
|
||||||
list_for_each_entry(lidp, &tp->t_items, lid_trans) {
|
list_for_each_entry(lidp, &tp->t_items, lid_trans) {
|
||||||
struct xfs_log_item *lip = lidp->lid_item;
|
struct xfs_log_item *lip = lidp->lid_item;
|
||||||
|
|
||||||
@ -441,44 +495,10 @@ xlog_cil_insert_items(
|
|||||||
list_move_tail(&lip->li_cil, &cil->xc_cil);
|
list_move_tail(&lip->li_cil, &cil->xc_cil);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* account for space used by new iovec headers */
|
|
||||||
len += diff_iovecs * sizeof(xlog_op_header_t);
|
|
||||||
ctx->nvecs += diff_iovecs;
|
|
||||||
|
|
||||||
/* attach the transaction to the CIL if it has any busy extents */
|
|
||||||
if (!list_empty(&tp->t_busy))
|
|
||||||
list_splice_init(&tp->t_busy, &ctx->busy_extents);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Now transfer enough transaction reservation to the context ticket
|
|
||||||
* for the checkpoint. The context ticket is special - the unit
|
|
||||||
* reservation has to grow as well as the current reservation as we
|
|
||||||
* steal from tickets so we can correctly determine the space used
|
|
||||||
* during the transaction commit.
|
|
||||||
*/
|
|
||||||
if (ctx->ticket->t_curr_res == 0) {
|
|
||||||
ctx->ticket->t_curr_res = ctx->ticket->t_unit_res;
|
|
||||||
tp->t_ticket->t_curr_res -= ctx->ticket->t_unit_res;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* do we need space for more log record headers? */
|
|
||||||
iclog_space = log->l_iclog_size - log->l_iclog_hsize;
|
|
||||||
if (len > 0 && (ctx->space_used / iclog_space !=
|
|
||||||
(ctx->space_used + len) / iclog_space)) {
|
|
||||||
int hdrs;
|
|
||||||
|
|
||||||
hdrs = (len + iclog_space - 1) / iclog_space;
|
|
||||||
/* need to take into account split region headers, too */
|
|
||||||
hdrs *= log->l_iclog_hsize + sizeof(struct xlog_op_header);
|
|
||||||
ctx->ticket->t_unit_res += hdrs;
|
|
||||||
ctx->ticket->t_curr_res += hdrs;
|
|
||||||
tp->t_ticket->t_curr_res -= hdrs;
|
|
||||||
ASSERT(tp->t_ticket->t_curr_res >= len);
|
|
||||||
}
|
|
||||||
tp->t_ticket->t_curr_res -= len;
|
|
||||||
ctx->space_used += len;
|
|
||||||
|
|
||||||
spin_unlock(&cil->xc_cil_lock);
|
spin_unlock(&cil->xc_cil_lock);
|
||||||
|
|
||||||
|
if (tp->t_ticket->t_curr_res < 0)
|
||||||
|
xfs_force_shutdown(log->l_mp, SHUTDOWN_LOG_IO_ERROR);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -973,6 +993,7 @@ xfs_log_commit_cil(
|
|||||||
{
|
{
|
||||||
struct xlog *log = mp->m_log;
|
struct xlog *log = mp->m_log;
|
||||||
struct xfs_cil *cil = log->l_cilp;
|
struct xfs_cil *cil = log->l_cilp;
|
||||||
|
xfs_lsn_t xc_commit_lsn;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Do all necessary memory allocation before we lock the CIL.
|
* Do all necessary memory allocation before we lock the CIL.
|
||||||
@ -986,13 +1007,9 @@ xfs_log_commit_cil(
|
|||||||
|
|
||||||
xlog_cil_insert_items(log, tp);
|
xlog_cil_insert_items(log, tp);
|
||||||
|
|
||||||
/* check we didn't blow the reservation */
|
xc_commit_lsn = cil->xc_ctx->sequence;
|
||||||
if (tp->t_ticket->t_curr_res < 0)
|
|
||||||
xlog_print_tic_res(mp, tp->t_ticket);
|
|
||||||
|
|
||||||
tp->t_commit_lsn = cil->xc_ctx->sequence;
|
|
||||||
if (commit_lsn)
|
if (commit_lsn)
|
||||||
*commit_lsn = tp->t_commit_lsn;
|
*commit_lsn = xc_commit_lsn;
|
||||||
|
|
||||||
xfs_log_done(mp, tp->t_ticket, NULL, regrant);
|
xfs_log_done(mp, tp->t_ticket, NULL, regrant);
|
||||||
xfs_trans_unreserve_and_mod_sb(tp);
|
xfs_trans_unreserve_and_mod_sb(tp);
|
||||||
@ -1008,7 +1025,7 @@ xfs_log_commit_cil(
|
|||||||
* the log items. This affects (at least) processing of stale buffers,
|
* the log items. This affects (at least) processing of stale buffers,
|
||||||
* inodes and EFIs.
|
* inodes and EFIs.
|
||||||
*/
|
*/
|
||||||
xfs_trans_free_items(tp, tp->t_commit_lsn, false);
|
xfs_trans_free_items(tp, xc_commit_lsn, false);
|
||||||
|
|
||||||
xlog_cil_push_background(log);
|
xlog_cil_push_background(log);
|
||||||
|
|
||||||
|
@ -419,7 +419,7 @@ struct xlog {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define XLOG_BUF_CANCEL_BUCKET(log, blkno) \
|
#define XLOG_BUF_CANCEL_BUCKET(log, blkno) \
|
||||||
((log)->l_buf_cancel_table + ((__uint64_t)blkno % XLOG_BC_TABLE_SIZE))
|
((log)->l_buf_cancel_table + ((uint64_t)blkno % XLOG_BC_TABLE_SIZE))
|
||||||
|
|
||||||
#define XLOG_FORCED_SHUTDOWN(log) ((log)->l_flags & XLOG_IO_ERROR)
|
#define XLOG_FORCED_SHUTDOWN(log) ((log)->l_flags & XLOG_IO_ERROR)
|
||||||
|
|
||||||
@ -456,6 +456,7 @@ xlog_write_adv_cnt(void **ptr, int *len, int *off, size_t bytes)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void xlog_print_tic_res(struct xfs_mount *mp, struct xlog_ticket *ticket);
|
void xlog_print_tic_res(struct xfs_mount *mp, struct xlog_ticket *ticket);
|
||||||
|
void xlog_print_trans(struct xfs_trans *);
|
||||||
int
|
int
|
||||||
xlog_write(
|
xlog_write(
|
||||||
struct xlog *log,
|
struct xlog *log,
|
||||||
|
@ -2230,9 +2230,9 @@ xlog_recover_get_buf_lsn(
|
|||||||
struct xfs_mount *mp,
|
struct xfs_mount *mp,
|
||||||
struct xfs_buf *bp)
|
struct xfs_buf *bp)
|
||||||
{
|
{
|
||||||
__uint32_t magic32;
|
uint32_t magic32;
|
||||||
__uint16_t magic16;
|
uint16_t magic16;
|
||||||
__uint16_t magicda;
|
uint16_t magicda;
|
||||||
void *blk = bp->b_addr;
|
void *blk = bp->b_addr;
|
||||||
uuid_t *uuid;
|
uuid_t *uuid;
|
||||||
xfs_lsn_t lsn = -1;
|
xfs_lsn_t lsn = -1;
|
||||||
@ -2381,9 +2381,9 @@ xlog_recover_validate_buf_type(
|
|||||||
xfs_lsn_t current_lsn)
|
xfs_lsn_t current_lsn)
|
||||||
{
|
{
|
||||||
struct xfs_da_blkinfo *info = bp->b_addr;
|
struct xfs_da_blkinfo *info = bp->b_addr;
|
||||||
__uint32_t magic32;
|
uint32_t magic32;
|
||||||
__uint16_t magic16;
|
uint16_t magic16;
|
||||||
__uint16_t magicda;
|
uint16_t magicda;
|
||||||
char *warnmsg = NULL;
|
char *warnmsg = NULL;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2852,7 +2852,7 @@ xlog_recover_buffer_pass2(
|
|||||||
if (XFS_DINODE_MAGIC ==
|
if (XFS_DINODE_MAGIC ==
|
||||||
be16_to_cpu(*((__be16 *)xfs_buf_offset(bp, 0))) &&
|
be16_to_cpu(*((__be16 *)xfs_buf_offset(bp, 0))) &&
|
||||||
(BBTOB(bp->b_io_length) != MAX(log->l_mp->m_sb.sb_blocksize,
|
(BBTOB(bp->b_io_length) != MAX(log->l_mp->m_sb.sb_blocksize,
|
||||||
(__uint32_t)log->l_mp->m_inode_cluster_size))) {
|
(uint32_t)log->l_mp->m_inode_cluster_size))) {
|
||||||
xfs_buf_stale(bp);
|
xfs_buf_stale(bp);
|
||||||
error = xfs_bwrite(bp);
|
error = xfs_bwrite(bp);
|
||||||
} else {
|
} else {
|
||||||
@ -3423,7 +3423,7 @@ xlog_recover_efd_pass2(
|
|||||||
xfs_efd_log_format_t *efd_formatp;
|
xfs_efd_log_format_t *efd_formatp;
|
||||||
xfs_efi_log_item_t *efip = NULL;
|
xfs_efi_log_item_t *efip = NULL;
|
||||||
xfs_log_item_t *lip;
|
xfs_log_item_t *lip;
|
||||||
__uint64_t efi_id;
|
uint64_t efi_id;
|
||||||
struct xfs_ail_cursor cur;
|
struct xfs_ail_cursor cur;
|
||||||
struct xfs_ail *ailp = log->l_ailp;
|
struct xfs_ail *ailp = log->l_ailp;
|
||||||
|
|
||||||
@ -3519,7 +3519,7 @@ xlog_recover_rud_pass2(
|
|||||||
struct xfs_rud_log_format *rud_formatp;
|
struct xfs_rud_log_format *rud_formatp;
|
||||||
struct xfs_rui_log_item *ruip = NULL;
|
struct xfs_rui_log_item *ruip = NULL;
|
||||||
struct xfs_log_item *lip;
|
struct xfs_log_item *lip;
|
||||||
__uint64_t rui_id;
|
uint64_t rui_id;
|
||||||
struct xfs_ail_cursor cur;
|
struct xfs_ail_cursor cur;
|
||||||
struct xfs_ail *ailp = log->l_ailp;
|
struct xfs_ail *ailp = log->l_ailp;
|
||||||
|
|
||||||
@ -3635,7 +3635,7 @@ xlog_recover_cud_pass2(
|
|||||||
struct xfs_cud_log_format *cud_formatp;
|
struct xfs_cud_log_format *cud_formatp;
|
||||||
struct xfs_cui_log_item *cuip = NULL;
|
struct xfs_cui_log_item *cuip = NULL;
|
||||||
struct xfs_log_item *lip;
|
struct xfs_log_item *lip;
|
||||||
__uint64_t cui_id;
|
uint64_t cui_id;
|
||||||
struct xfs_ail_cursor cur;
|
struct xfs_ail_cursor cur;
|
||||||
struct xfs_ail *ailp = log->l_ailp;
|
struct xfs_ail *ailp = log->l_ailp;
|
||||||
|
|
||||||
@ -3754,7 +3754,7 @@ xlog_recover_bud_pass2(
|
|||||||
struct xfs_bud_log_format *bud_formatp;
|
struct xfs_bud_log_format *bud_formatp;
|
||||||
struct xfs_bui_log_item *buip = NULL;
|
struct xfs_bui_log_item *buip = NULL;
|
||||||
struct xfs_log_item *lip;
|
struct xfs_log_item *lip;
|
||||||
__uint64_t bui_id;
|
uint64_t bui_id;
|
||||||
struct xfs_ail_cursor cur;
|
struct xfs_ail_cursor cur;
|
||||||
struct xfs_ail *ailp = log->l_ailp;
|
struct xfs_ail *ailp = log->l_ailp;
|
||||||
|
|
||||||
@ -4152,7 +4152,7 @@ xlog_recover_commit_trans(
|
|||||||
|
|
||||||
#define XLOG_RECOVER_COMMIT_QUEUE_MAX 100
|
#define XLOG_RECOVER_COMMIT_QUEUE_MAX 100
|
||||||
|
|
||||||
hlist_del(&trans->r_list);
|
hlist_del_init(&trans->r_list);
|
||||||
|
|
||||||
error = xlog_recover_reorder_trans(log, trans, pass);
|
error = xlog_recover_reorder_trans(log, trans, pass);
|
||||||
if (error)
|
if (error)
|
||||||
@ -4354,6 +4354,8 @@ xlog_recover_free_trans(
|
|||||||
xlog_recover_item_t *item, *n;
|
xlog_recover_item_t *item, *n;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
hlist_del_init(&trans->r_list);
|
||||||
|
|
||||||
list_for_each_entry_safe(item, n, &trans->r_itemq, ri_list) {
|
list_for_each_entry_safe(item, n, &trans->r_itemq, ri_list) {
|
||||||
/* Free the regions in the item. */
|
/* Free the regions in the item. */
|
||||||
list_del(&item->ri_list);
|
list_del(&item->ri_list);
|
||||||
@ -5224,12 +5226,16 @@ xlog_do_recovery_pass(
|
|||||||
int error2 = 0;
|
int error2 = 0;
|
||||||
int bblks, split_bblks;
|
int bblks, split_bblks;
|
||||||
int hblks, split_hblks, wrapped_hblks;
|
int hblks, split_hblks, wrapped_hblks;
|
||||||
|
int i;
|
||||||
struct hlist_head rhash[XLOG_RHASH_SIZE];
|
struct hlist_head rhash[XLOG_RHASH_SIZE];
|
||||||
LIST_HEAD (buffer_list);
|
LIST_HEAD (buffer_list);
|
||||||
|
|
||||||
ASSERT(head_blk != tail_blk);
|
ASSERT(head_blk != tail_blk);
|
||||||
rhead_blk = 0;
|
rhead_blk = 0;
|
||||||
|
|
||||||
|
for (i = 0; i < XLOG_RHASH_SIZE; i++)
|
||||||
|
INIT_HLIST_HEAD(&rhash[i]);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read the header of the tail block and get the iclog buffer size from
|
* Read the header of the tail block and get the iclog buffer size from
|
||||||
* h_size. Use this to tell how many sectors make up the log header.
|
* h_size. Use this to tell how many sectors make up the log header.
|
||||||
@ -5466,6 +5472,19 @@ xlog_do_recovery_pass(
|
|||||||
if (error && first_bad)
|
if (error && first_bad)
|
||||||
*first_bad = rhead_blk;
|
*first_bad = rhead_blk;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Transactions are freed at commit time but transactions without commit
|
||||||
|
* records on disk are never committed. Free any that may be left in the
|
||||||
|
* hash table.
|
||||||
|
*/
|
||||||
|
for (i = 0; i < XLOG_RHASH_SIZE; i++) {
|
||||||
|
struct hlist_node *tmp;
|
||||||
|
struct xlog_recover *trans;
|
||||||
|
|
||||||
|
hlist_for_each_entry_safe(trans, tmp, &rhash[i], r_list)
|
||||||
|
xlog_recover_free_trans(trans);
|
||||||
|
}
|
||||||
|
|
||||||
return error ? error : error2;
|
return error ? error : error2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5772,9 +5791,9 @@ xlog_recover_check_summary(
|
|||||||
xfs_buf_t *agfbp;
|
xfs_buf_t *agfbp;
|
||||||
xfs_buf_t *agibp;
|
xfs_buf_t *agibp;
|
||||||
xfs_agnumber_t agno;
|
xfs_agnumber_t agno;
|
||||||
__uint64_t freeblks;
|
uint64_t freeblks;
|
||||||
__uint64_t itotal;
|
uint64_t itotal;
|
||||||
__uint64_t ifree;
|
uint64_t ifree;
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
mp = log->l_mp;
|
mp = log->l_mp;
|
||||||
|
@ -110,7 +110,10 @@ assfail(char *expr, char *file, int line)
|
|||||||
{
|
{
|
||||||
xfs_emerg(NULL, "Assertion failed: %s, file: %s, line: %d",
|
xfs_emerg(NULL, "Assertion failed: %s, file: %s, line: %d",
|
||||||
expr, file, line);
|
expr, file, line);
|
||||||
BUG();
|
if (xfs_globals.bug_on_assert)
|
||||||
|
BUG();
|
||||||
|
else
|
||||||
|
WARN_ON(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -173,7 +173,7 @@ xfs_free_perag(
|
|||||||
int
|
int
|
||||||
xfs_sb_validate_fsb_count(
|
xfs_sb_validate_fsb_count(
|
||||||
xfs_sb_t *sbp,
|
xfs_sb_t *sbp,
|
||||||
__uint64_t nblocks)
|
uint64_t nblocks)
|
||||||
{
|
{
|
||||||
ASSERT(PAGE_SHIFT >= sbp->sb_blocklog);
|
ASSERT(PAGE_SHIFT >= sbp->sb_blocklog);
|
||||||
ASSERT(sbp->sb_blocklog >= BBSHIFT);
|
ASSERT(sbp->sb_blocklog >= BBSHIFT);
|
||||||
@ -435,7 +435,7 @@ STATIC void
|
|||||||
xfs_set_maxicount(xfs_mount_t *mp)
|
xfs_set_maxicount(xfs_mount_t *mp)
|
||||||
{
|
{
|
||||||
xfs_sb_t *sbp = &(mp->m_sb);
|
xfs_sb_t *sbp = &(mp->m_sb);
|
||||||
__uint64_t icount;
|
uint64_t icount;
|
||||||
|
|
||||||
if (sbp->sb_imax_pct) {
|
if (sbp->sb_imax_pct) {
|
||||||
/*
|
/*
|
||||||
@ -501,7 +501,7 @@ xfs_set_low_space_thresholds(
|
|||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < XFS_LOWSP_MAX; i++) {
|
for (i = 0; i < XFS_LOWSP_MAX; i++) {
|
||||||
__uint64_t space = mp->m_sb.sb_dblocks;
|
uint64_t space = mp->m_sb.sb_dblocks;
|
||||||
|
|
||||||
do_div(space, 100);
|
do_div(space, 100);
|
||||||
mp->m_low_space[i] = space * (i + 1);
|
mp->m_low_space[i] = space * (i + 1);
|
||||||
@ -597,10 +597,10 @@ xfs_mount_reset_sbqflags(
|
|||||||
return xfs_sync_sb(mp, false);
|
return xfs_sync_sb(mp, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
__uint64_t
|
uint64_t
|
||||||
xfs_default_resblks(xfs_mount_t *mp)
|
xfs_default_resblks(xfs_mount_t *mp)
|
||||||
{
|
{
|
||||||
__uint64_t resblks;
|
uint64_t resblks;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We default to 5% or 8192 fsbs of space reserved, whichever is
|
* We default to 5% or 8192 fsbs of space reserved, whichever is
|
||||||
@ -611,7 +611,7 @@ xfs_default_resblks(xfs_mount_t *mp)
|
|||||||
*/
|
*/
|
||||||
resblks = mp->m_sb.sb_dblocks;
|
resblks = mp->m_sb.sb_dblocks;
|
||||||
do_div(resblks, 20);
|
do_div(resblks, 20);
|
||||||
resblks = min_t(__uint64_t, resblks, 8192);
|
resblks = min_t(uint64_t, resblks, 8192);
|
||||||
return resblks;
|
return resblks;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -631,7 +631,7 @@ xfs_mountfs(
|
|||||||
{
|
{
|
||||||
struct xfs_sb *sbp = &(mp->m_sb);
|
struct xfs_sb *sbp = &(mp->m_sb);
|
||||||
struct xfs_inode *rip;
|
struct xfs_inode *rip;
|
||||||
__uint64_t resblks;
|
uint64_t resblks;
|
||||||
uint quotamount = 0;
|
uint quotamount = 0;
|
||||||
uint quotaflags = 0;
|
uint quotaflags = 0;
|
||||||
int error = 0;
|
int error = 0;
|
||||||
@ -719,10 +719,13 @@ xfs_mountfs(
|
|||||||
if (error)
|
if (error)
|
||||||
goto out_del_stats;
|
goto out_del_stats;
|
||||||
|
|
||||||
|
error = xfs_errortag_init(mp);
|
||||||
|
if (error)
|
||||||
|
goto out_remove_error_sysfs;
|
||||||
|
|
||||||
error = xfs_uuid_mount(mp);
|
error = xfs_uuid_mount(mp);
|
||||||
if (error)
|
if (error)
|
||||||
goto out_remove_error_sysfs;
|
goto out_remove_errortag;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Set the minimum read and write sizes
|
* Set the minimum read and write sizes
|
||||||
@ -1044,6 +1047,8 @@ xfs_mountfs(
|
|||||||
xfs_da_unmount(mp);
|
xfs_da_unmount(mp);
|
||||||
out_remove_uuid:
|
out_remove_uuid:
|
||||||
xfs_uuid_unmount(mp);
|
xfs_uuid_unmount(mp);
|
||||||
|
out_remove_errortag:
|
||||||
|
xfs_errortag_del(mp);
|
||||||
out_remove_error_sysfs:
|
out_remove_error_sysfs:
|
||||||
xfs_error_sysfs_del(mp);
|
xfs_error_sysfs_del(mp);
|
||||||
out_del_stats:
|
out_del_stats:
|
||||||
@ -1062,7 +1067,7 @@ void
|
|||||||
xfs_unmountfs(
|
xfs_unmountfs(
|
||||||
struct xfs_mount *mp)
|
struct xfs_mount *mp)
|
||||||
{
|
{
|
||||||
__uint64_t resblks;
|
uint64_t resblks;
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
cancel_delayed_work_sync(&mp->m_eofblocks_work);
|
cancel_delayed_work_sync(&mp->m_eofblocks_work);
|
||||||
@ -1147,10 +1152,11 @@ xfs_unmountfs(
|
|||||||
xfs_uuid_unmount(mp);
|
xfs_uuid_unmount(mp);
|
||||||
|
|
||||||
#if defined(DEBUG)
|
#if defined(DEBUG)
|
||||||
xfs_errortag_clearall(mp, 0);
|
xfs_errortag_clearall(mp);
|
||||||
#endif
|
#endif
|
||||||
xfs_free_perag(mp);
|
xfs_free_perag(mp);
|
||||||
|
|
||||||
|
xfs_errortag_del(mp);
|
||||||
xfs_error_sysfs_del(mp);
|
xfs_error_sysfs_del(mp);
|
||||||
xfs_sysfs_del(&mp->m_stats.xs_kobj);
|
xfs_sysfs_del(&mp->m_stats.xs_kobj);
|
||||||
xfs_sysfs_del(&mp->m_kobj);
|
xfs_sysfs_del(&mp->m_kobj);
|
||||||
|
@ -108,10 +108,10 @@ typedef struct xfs_mount {
|
|||||||
xfs_buftarg_t *m_ddev_targp; /* saves taking the address */
|
xfs_buftarg_t *m_ddev_targp; /* saves taking the address */
|
||||||
xfs_buftarg_t *m_logdev_targp;/* ptr to log device */
|
xfs_buftarg_t *m_logdev_targp;/* ptr to log device */
|
||||||
xfs_buftarg_t *m_rtdev_targp; /* ptr to rt device */
|
xfs_buftarg_t *m_rtdev_targp; /* ptr to rt device */
|
||||||
__uint8_t m_blkbit_log; /* blocklog + NBBY */
|
uint8_t m_blkbit_log; /* blocklog + NBBY */
|
||||||
__uint8_t m_blkbb_log; /* blocklog - BBSHIFT */
|
uint8_t m_blkbb_log; /* blocklog - BBSHIFT */
|
||||||
__uint8_t m_agno_log; /* log #ag's */
|
uint8_t m_agno_log; /* log #ag's */
|
||||||
__uint8_t m_agino_log; /* #bits for agino in inum */
|
uint8_t m_agino_log; /* #bits for agino in inum */
|
||||||
uint m_inode_cluster_size;/* min inode buf size */
|
uint m_inode_cluster_size;/* min inode buf size */
|
||||||
uint m_blockmask; /* sb_blocksize-1 */
|
uint m_blockmask; /* sb_blocksize-1 */
|
||||||
uint m_blockwsize; /* sb_blocksize in words */
|
uint m_blockwsize; /* sb_blocksize in words */
|
||||||
@ -139,7 +139,7 @@ typedef struct xfs_mount {
|
|||||||
struct mutex m_growlock; /* growfs mutex */
|
struct mutex m_growlock; /* growfs mutex */
|
||||||
int m_fixedfsid[2]; /* unchanged for life of FS */
|
int m_fixedfsid[2]; /* unchanged for life of FS */
|
||||||
uint m_dmevmask; /* DMI events for this FS */
|
uint m_dmevmask; /* DMI events for this FS */
|
||||||
__uint64_t m_flags; /* global mount flags */
|
uint64_t m_flags; /* global mount flags */
|
||||||
bool m_inotbt_nores; /* no per-AG finobt resv. */
|
bool m_inotbt_nores; /* no per-AG finobt resv. */
|
||||||
int m_ialloc_inos; /* inodes in inode allocation */
|
int m_ialloc_inos; /* inodes in inode allocation */
|
||||||
int m_ialloc_blks; /* blocks in inode allocation */
|
int m_ialloc_blks; /* blocks in inode allocation */
|
||||||
@ -148,14 +148,14 @@ typedef struct xfs_mount {
|
|||||||
int m_inoalign_mask;/* mask sb_inoalignmt if used */
|
int m_inoalign_mask;/* mask sb_inoalignmt if used */
|
||||||
uint m_qflags; /* quota status flags */
|
uint m_qflags; /* quota status flags */
|
||||||
struct xfs_trans_resv m_resv; /* precomputed res values */
|
struct xfs_trans_resv m_resv; /* precomputed res values */
|
||||||
__uint64_t m_maxicount; /* maximum inode count */
|
uint64_t m_maxicount; /* maximum inode count */
|
||||||
__uint64_t m_resblks; /* total reserved blocks */
|
uint64_t m_resblks; /* total reserved blocks */
|
||||||
__uint64_t m_resblks_avail;/* available reserved blocks */
|
uint64_t m_resblks_avail;/* available reserved blocks */
|
||||||
__uint64_t m_resblks_save; /* reserved blks @ remount,ro */
|
uint64_t m_resblks_save; /* reserved blks @ remount,ro */
|
||||||
int m_dalign; /* stripe unit */
|
int m_dalign; /* stripe unit */
|
||||||
int m_swidth; /* stripe width */
|
int m_swidth; /* stripe width */
|
||||||
int m_sinoalign; /* stripe unit inode alignment */
|
int m_sinoalign; /* stripe unit inode alignment */
|
||||||
__uint8_t m_sectbb_log; /* sectlog - BBSHIFT */
|
uint8_t m_sectbb_log; /* sectlog - BBSHIFT */
|
||||||
const struct xfs_nameops *m_dirnameops; /* vector of dir name ops */
|
const struct xfs_nameops *m_dirnameops; /* vector of dir name ops */
|
||||||
const struct xfs_dir_ops *m_dir_inode_ops; /* vector of dir inode ops */
|
const struct xfs_dir_ops *m_dir_inode_ops; /* vector of dir inode ops */
|
||||||
const struct xfs_dir_ops *m_nondir_inode_ops; /* !dir inode ops */
|
const struct xfs_dir_ops *m_nondir_inode_ops; /* !dir inode ops */
|
||||||
@ -194,19 +194,17 @@ typedef struct xfs_mount {
|
|||||||
* ever support shrinks it would have to be persisted in addition
|
* ever support shrinks it would have to be persisted in addition
|
||||||
* to various other kinds of pain inflicted on the pNFS server.
|
* to various other kinds of pain inflicted on the pNFS server.
|
||||||
*/
|
*/
|
||||||
__uint32_t m_generation;
|
uint32_t m_generation;
|
||||||
|
|
||||||
bool m_fail_unmount;
|
bool m_fail_unmount;
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
/*
|
/*
|
||||||
* DEBUG mode instrumentation to test and/or trigger delayed allocation
|
* Frequency with which errors are injected. Replaces xfs_etest; the
|
||||||
* block killing in the event of failed writes. When enabled, all
|
* value stored in here is the inverse of the frequency with which the
|
||||||
* buffered writes are silenty dropped and handled as if they failed.
|
* error triggers. 1 = always, 2 = half the time, etc.
|
||||||
* All delalloc blocks in the range of the write (including pre-existing
|
|
||||||
* delalloc blocks!) are tossed as part of the write failure error
|
|
||||||
* handling sequence.
|
|
||||||
*/
|
*/
|
||||||
bool m_drop_writes;
|
unsigned int *m_errortag;
|
||||||
|
struct xfs_kobj m_errortag_kobj;
|
||||||
#endif
|
#endif
|
||||||
} xfs_mount_t;
|
} xfs_mount_t;
|
||||||
|
|
||||||
@ -325,20 +323,6 @@ xfs_daddr_to_agbno(struct xfs_mount *mp, xfs_daddr_t d)
|
|||||||
return (xfs_agblock_t) do_div(ld, mp->m_sb.sb_agblocks);
|
return (xfs_agblock_t) do_div(ld, mp->m_sb.sb_agblocks);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG
|
|
||||||
static inline bool
|
|
||||||
xfs_mp_drop_writes(struct xfs_mount *mp)
|
|
||||||
{
|
|
||||||
return mp->m_drop_writes;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
static inline bool
|
|
||||||
xfs_mp_drop_writes(struct xfs_mount *mp)
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* per-AG block reservation data structures*/
|
/* per-AG block reservation data structures*/
|
||||||
enum xfs_ag_resv_type {
|
enum xfs_ag_resv_type {
|
||||||
XFS_AG_RESV_NONE = 0,
|
XFS_AG_RESV_NONE = 0,
|
||||||
@ -367,12 +351,12 @@ typedef struct xfs_perag {
|
|||||||
char pagi_init; /* this agi's entry is initialized */
|
char pagi_init; /* this agi's entry is initialized */
|
||||||
char pagf_metadata; /* the agf is preferred to be metadata */
|
char pagf_metadata; /* the agf is preferred to be metadata */
|
||||||
char pagi_inodeok; /* The agi is ok for inodes */
|
char pagi_inodeok; /* The agi is ok for inodes */
|
||||||
__uint8_t pagf_levels[XFS_BTNUM_AGF];
|
uint8_t pagf_levels[XFS_BTNUM_AGF];
|
||||||
/* # of levels in bno & cnt btree */
|
/* # of levels in bno & cnt btree */
|
||||||
__uint32_t pagf_flcount; /* count of blocks in freelist */
|
uint32_t pagf_flcount; /* count of blocks in freelist */
|
||||||
xfs_extlen_t pagf_freeblks; /* total free blocks */
|
xfs_extlen_t pagf_freeblks; /* total free blocks */
|
||||||
xfs_extlen_t pagf_longest; /* longest free space */
|
xfs_extlen_t pagf_longest; /* longest free space */
|
||||||
__uint32_t pagf_btreeblks; /* # of blocks held in AGF btrees */
|
uint32_t pagf_btreeblks; /* # of blocks held in AGF btrees */
|
||||||
xfs_agino_t pagi_freecount; /* number of free inodes */
|
xfs_agino_t pagi_freecount; /* number of free inodes */
|
||||||
xfs_agino_t pagi_count; /* number of allocated inodes */
|
xfs_agino_t pagi_count; /* number of allocated inodes */
|
||||||
|
|
||||||
@ -411,7 +395,7 @@ typedef struct xfs_perag {
|
|||||||
struct xfs_ag_resv pag_agfl_resv;
|
struct xfs_ag_resv pag_agfl_resv;
|
||||||
|
|
||||||
/* reference count */
|
/* reference count */
|
||||||
__uint8_t pagf_refcount_level;
|
uint8_t pagf_refcount_level;
|
||||||
} xfs_perag_t;
|
} xfs_perag_t;
|
||||||
|
|
||||||
static inline struct xfs_ag_resv *
|
static inline struct xfs_ag_resv *
|
||||||
@ -434,7 +418,7 @@ void xfs_buf_hash_destroy(xfs_perag_t *pag);
|
|||||||
|
|
||||||
extern void xfs_uuid_table_free(void);
|
extern void xfs_uuid_table_free(void);
|
||||||
extern int xfs_log_sbcount(xfs_mount_t *);
|
extern int xfs_log_sbcount(xfs_mount_t *);
|
||||||
extern __uint64_t xfs_default_resblks(xfs_mount_t *mp);
|
extern uint64_t xfs_default_resblks(xfs_mount_t *mp);
|
||||||
extern int xfs_mountfs(xfs_mount_t *mp);
|
extern int xfs_mountfs(xfs_mount_t *mp);
|
||||||
extern int xfs_initialize_perag(xfs_mount_t *mp, xfs_agnumber_t agcount,
|
extern int xfs_initialize_perag(xfs_mount_t *mp, xfs_agnumber_t agcount,
|
||||||
xfs_agnumber_t *maxagi);
|
xfs_agnumber_t *maxagi);
|
||||||
@ -450,7 +434,7 @@ extern struct xfs_buf *xfs_getsb(xfs_mount_t *, int);
|
|||||||
extern int xfs_readsb(xfs_mount_t *, int);
|
extern int xfs_readsb(xfs_mount_t *, int);
|
||||||
extern void xfs_freesb(xfs_mount_t *);
|
extern void xfs_freesb(xfs_mount_t *);
|
||||||
extern bool xfs_fs_writable(struct xfs_mount *mp, int level);
|
extern bool xfs_fs_writable(struct xfs_mount *mp, int level);
|
||||||
extern int xfs_sb_validate_fsb_count(struct xfs_sb *, __uint64_t);
|
extern int xfs_sb_validate_fsb_count(struct xfs_sb *, uint64_t);
|
||||||
|
|
||||||
extern int xfs_dev_is_read_only(struct xfs_mount *, char *);
|
extern int xfs_dev_is_read_only(struct xfs_mount *, char *);
|
||||||
|
|
||||||
|
@ -1247,6 +1247,7 @@ xfs_qm_flush_one(
|
|||||||
struct xfs_dquot *dqp,
|
struct xfs_dquot *dqp,
|
||||||
void *data)
|
void *data)
|
||||||
{
|
{
|
||||||
|
struct xfs_mount *mp = dqp->q_mount;
|
||||||
struct list_head *buffer_list = data;
|
struct list_head *buffer_list = data;
|
||||||
struct xfs_buf *bp = NULL;
|
struct xfs_buf *bp = NULL;
|
||||||
int error = 0;
|
int error = 0;
|
||||||
@ -1257,7 +1258,32 @@ xfs_qm_flush_one(
|
|||||||
if (!XFS_DQ_IS_DIRTY(dqp))
|
if (!XFS_DQ_IS_DIRTY(dqp))
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
|
|
||||||
xfs_dqflock(dqp);
|
/*
|
||||||
|
* The only way the dquot is already flush locked by the time quotacheck
|
||||||
|
* gets here is if reclaim flushed it before the dqadjust walk dirtied
|
||||||
|
* it for the final time. Quotacheck collects all dquot bufs in the
|
||||||
|
* local delwri queue before dquots are dirtied, so reclaim can't have
|
||||||
|
* possibly queued it for I/O. The only way out is to push the buffer to
|
||||||
|
* cycle the flush lock.
|
||||||
|
*/
|
||||||
|
if (!xfs_dqflock_nowait(dqp)) {
|
||||||
|
/* buf is pinned in-core by delwri list */
|
||||||
|
DEFINE_SINGLE_BUF_MAP(map, dqp->q_blkno,
|
||||||
|
mp->m_quotainfo->qi_dqchunklen);
|
||||||
|
bp = _xfs_buf_find(mp->m_ddev_targp, &map, 1, 0, NULL);
|
||||||
|
if (!bp) {
|
||||||
|
error = -EINVAL;
|
||||||
|
goto out_unlock;
|
||||||
|
}
|
||||||
|
xfs_buf_unlock(bp);
|
||||||
|
|
||||||
|
xfs_buf_delwri_pushbuf(bp, buffer_list);
|
||||||
|
xfs_buf_rele(bp);
|
||||||
|
|
||||||
|
error = -EAGAIN;
|
||||||
|
goto out_unlock;
|
||||||
|
}
|
||||||
|
|
||||||
error = xfs_qm_dqflush(dqp, &bp);
|
error = xfs_qm_dqflush(dqp, &bp);
|
||||||
if (error)
|
if (error)
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
|
@ -33,7 +33,7 @@ xfs_fill_statvfs_from_dquot(
|
|||||||
struct kstatfs *statp,
|
struct kstatfs *statp,
|
||||||
struct xfs_dquot *dqp)
|
struct xfs_dquot *dqp)
|
||||||
{
|
{
|
||||||
__uint64_t limit;
|
uint64_t limit;
|
||||||
|
|
||||||
limit = dqp->q_core.d_blk_softlimit ?
|
limit = dqp->q_core.d_blk_softlimit ?
|
||||||
be64_to_cpu(dqp->q_core.d_blk_softlimit) :
|
be64_to_cpu(dqp->q_core.d_blk_softlimit) :
|
||||||
|
@ -269,7 +269,6 @@ xfs_fs_get_nextdqblk(
|
|||||||
/* ID may be different, so convert back what we got */
|
/* ID may be different, so convert back what we got */
|
||||||
*qid = make_kqid(current_user_ns(), qid->type, id);
|
*qid = make_kqid(current_user_ns(), qid->type, id);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC int
|
STATIC int
|
||||||
|
@ -155,6 +155,7 @@
|
|||||||
int
|
int
|
||||||
xfs_reflink_find_shared(
|
xfs_reflink_find_shared(
|
||||||
struct xfs_mount *mp,
|
struct xfs_mount *mp,
|
||||||
|
struct xfs_trans *tp,
|
||||||
xfs_agnumber_t agno,
|
xfs_agnumber_t agno,
|
||||||
xfs_agblock_t agbno,
|
xfs_agblock_t agbno,
|
||||||
xfs_extlen_t aglen,
|
xfs_extlen_t aglen,
|
||||||
@ -166,18 +167,18 @@ xfs_reflink_find_shared(
|
|||||||
struct xfs_btree_cur *cur;
|
struct xfs_btree_cur *cur;
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
error = xfs_alloc_read_agf(mp, NULL, agno, 0, &agbp);
|
error = xfs_alloc_read_agf(mp, tp, agno, 0, &agbp);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
|
|
||||||
cur = xfs_refcountbt_init_cursor(mp, NULL, agbp, agno, NULL);
|
cur = xfs_refcountbt_init_cursor(mp, tp, agbp, agno, NULL);
|
||||||
|
|
||||||
error = xfs_refcount_find_shared(cur, agbno, aglen, fbno, flen,
|
error = xfs_refcount_find_shared(cur, agbno, aglen, fbno, flen,
|
||||||
find_end_of_shared);
|
find_end_of_shared);
|
||||||
|
|
||||||
xfs_btree_del_cursor(cur, error ? XFS_BTREE_ERROR : XFS_BTREE_NOERROR);
|
xfs_btree_del_cursor(cur, error ? XFS_BTREE_ERROR : XFS_BTREE_NOERROR);
|
||||||
|
|
||||||
xfs_buf_relse(agbp);
|
xfs_trans_brelse(tp, agbp);
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -217,7 +218,7 @@ xfs_reflink_trim_around_shared(
|
|||||||
agbno = XFS_FSB_TO_AGBNO(ip->i_mount, irec->br_startblock);
|
agbno = XFS_FSB_TO_AGBNO(ip->i_mount, irec->br_startblock);
|
||||||
aglen = irec->br_blockcount;
|
aglen = irec->br_blockcount;
|
||||||
|
|
||||||
error = xfs_reflink_find_shared(ip->i_mount, agno, agbno,
|
error = xfs_reflink_find_shared(ip->i_mount, NULL, agno, agbno,
|
||||||
aglen, &fbno, &flen, true);
|
aglen, &fbno, &flen, true);
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
@ -1373,8 +1374,8 @@ xfs_reflink_dirty_extents(
|
|||||||
agbno = XFS_FSB_TO_AGBNO(mp, map[1].br_startblock);
|
agbno = XFS_FSB_TO_AGBNO(mp, map[1].br_startblock);
|
||||||
aglen = map[1].br_blockcount;
|
aglen = map[1].br_blockcount;
|
||||||
|
|
||||||
error = xfs_reflink_find_shared(mp, agno, agbno, aglen,
|
error = xfs_reflink_find_shared(mp, NULL, agno, agbno,
|
||||||
&rbno, &rlen, true);
|
aglen, &rbno, &rlen, true);
|
||||||
if (error)
|
if (error)
|
||||||
goto out;
|
goto out;
|
||||||
if (rbno == NULLAGBLOCK)
|
if (rbno == NULLAGBLOCK)
|
||||||
@ -1405,56 +1406,72 @@ out:
|
|||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Does this inode need the reflink flag? */
|
||||||
|
int
|
||||||
|
xfs_reflink_inode_has_shared_extents(
|
||||||
|
struct xfs_trans *tp,
|
||||||
|
struct xfs_inode *ip,
|
||||||
|
bool *has_shared)
|
||||||
|
{
|
||||||
|
struct xfs_bmbt_irec got;
|
||||||
|
struct xfs_mount *mp = ip->i_mount;
|
||||||
|
struct xfs_ifork *ifp;
|
||||||
|
xfs_agnumber_t agno;
|
||||||
|
xfs_agblock_t agbno;
|
||||||
|
xfs_extlen_t aglen;
|
||||||
|
xfs_agblock_t rbno;
|
||||||
|
xfs_extlen_t rlen;
|
||||||
|
xfs_extnum_t idx;
|
||||||
|
bool found;
|
||||||
|
int error;
|
||||||
|
|
||||||
|
ifp = XFS_IFORK_PTR(ip, XFS_DATA_FORK);
|
||||||
|
if (!(ifp->if_flags & XFS_IFEXTENTS)) {
|
||||||
|
error = xfs_iread_extents(tp, ip, XFS_DATA_FORK);
|
||||||
|
if (error)
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
*has_shared = false;
|
||||||
|
found = xfs_iext_lookup_extent(ip, ifp, 0, &idx, &got);
|
||||||
|
while (found) {
|
||||||
|
if (isnullstartblock(got.br_startblock) ||
|
||||||
|
got.br_state != XFS_EXT_NORM)
|
||||||
|
goto next;
|
||||||
|
agno = XFS_FSB_TO_AGNO(mp, got.br_startblock);
|
||||||
|
agbno = XFS_FSB_TO_AGBNO(mp, got.br_startblock);
|
||||||
|
aglen = got.br_blockcount;
|
||||||
|
|
||||||
|
error = xfs_reflink_find_shared(mp, tp, agno, agbno, aglen,
|
||||||
|
&rbno, &rlen, false);
|
||||||
|
if (error)
|
||||||
|
return error;
|
||||||
|
/* Is there still a shared block here? */
|
||||||
|
if (rbno != NULLAGBLOCK) {
|
||||||
|
*has_shared = true;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
next:
|
||||||
|
found = xfs_iext_get_extent(ifp, ++idx, &got);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* Clear the inode reflink flag if there are no shared extents. */
|
/* Clear the inode reflink flag if there are no shared extents. */
|
||||||
int
|
int
|
||||||
xfs_reflink_clear_inode_flag(
|
xfs_reflink_clear_inode_flag(
|
||||||
struct xfs_inode *ip,
|
struct xfs_inode *ip,
|
||||||
struct xfs_trans **tpp)
|
struct xfs_trans **tpp)
|
||||||
{
|
{
|
||||||
struct xfs_mount *mp = ip->i_mount;
|
bool needs_flag;
|
||||||
xfs_fileoff_t fbno;
|
|
||||||
xfs_filblks_t end;
|
|
||||||
xfs_agnumber_t agno;
|
|
||||||
xfs_agblock_t agbno;
|
|
||||||
xfs_extlen_t aglen;
|
|
||||||
xfs_agblock_t rbno;
|
|
||||||
xfs_extlen_t rlen;
|
|
||||||
struct xfs_bmbt_irec map;
|
|
||||||
int nmaps;
|
|
||||||
int error = 0;
|
int error = 0;
|
||||||
|
|
||||||
ASSERT(xfs_is_reflink_inode(ip));
|
ASSERT(xfs_is_reflink_inode(ip));
|
||||||
|
|
||||||
fbno = 0;
|
error = xfs_reflink_inode_has_shared_extents(*tpp, ip, &needs_flag);
|
||||||
end = XFS_B_TO_FSB(mp, i_size_read(VFS_I(ip)));
|
if (error || needs_flag)
|
||||||
while (end - fbno > 0) {
|
return error;
|
||||||
nmaps = 1;
|
|
||||||
/*
|
|
||||||
* Look for extents in the file. Skip holes, delalloc, or
|
|
||||||
* unwritten extents; they can't be reflinked.
|
|
||||||
*/
|
|
||||||
error = xfs_bmapi_read(ip, fbno, end - fbno, &map, &nmaps, 0);
|
|
||||||
if (error)
|
|
||||||
return error;
|
|
||||||
if (nmaps == 0)
|
|
||||||
break;
|
|
||||||
if (!xfs_bmap_is_real_extent(&map))
|
|
||||||
goto next;
|
|
||||||
|
|
||||||
agno = XFS_FSB_TO_AGNO(mp, map.br_startblock);
|
|
||||||
agbno = XFS_FSB_TO_AGBNO(mp, map.br_startblock);
|
|
||||||
aglen = map.br_blockcount;
|
|
||||||
|
|
||||||
error = xfs_reflink_find_shared(mp, agno, agbno, aglen,
|
|
||||||
&rbno, &rlen, false);
|
|
||||||
if (error)
|
|
||||||
return error;
|
|
||||||
/* Is there still a shared block here? */
|
|
||||||
if (rbno != NULLAGBLOCK)
|
|
||||||
return 0;
|
|
||||||
next:
|
|
||||||
fbno = map.br_startoff + map.br_blockcount;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We didn't find any shared blocks so turn off the reflink flag.
|
* We didn't find any shared blocks so turn off the reflink flag.
|
||||||
|
@ -20,9 +20,9 @@
|
|||||||
#ifndef __XFS_REFLINK_H
|
#ifndef __XFS_REFLINK_H
|
||||||
#define __XFS_REFLINK_H 1
|
#define __XFS_REFLINK_H 1
|
||||||
|
|
||||||
extern int xfs_reflink_find_shared(struct xfs_mount *mp, xfs_agnumber_t agno,
|
extern int xfs_reflink_find_shared(struct xfs_mount *mp, struct xfs_trans *tp,
|
||||||
xfs_agblock_t agbno, xfs_extlen_t aglen, xfs_agblock_t *fbno,
|
xfs_agnumber_t agno, xfs_agblock_t agbno, xfs_extlen_t aglen,
|
||||||
xfs_extlen_t *flen, bool find_maximal);
|
xfs_agblock_t *fbno, xfs_extlen_t *flen, bool find_maximal);
|
||||||
extern int xfs_reflink_trim_around_shared(struct xfs_inode *ip,
|
extern int xfs_reflink_trim_around_shared(struct xfs_inode *ip,
|
||||||
struct xfs_bmbt_irec *irec, bool *shared, bool *trimmed);
|
struct xfs_bmbt_irec *irec, bool *shared, bool *trimmed);
|
||||||
|
|
||||||
@ -47,6 +47,8 @@ extern int xfs_reflink_end_cow(struct xfs_inode *ip, xfs_off_t offset,
|
|||||||
extern int xfs_reflink_recover_cow(struct xfs_mount *mp);
|
extern int xfs_reflink_recover_cow(struct xfs_mount *mp);
|
||||||
extern int xfs_reflink_remap_range(struct file *file_in, loff_t pos_in,
|
extern int xfs_reflink_remap_range(struct file *file_in, loff_t pos_in,
|
||||||
struct file *file_out, loff_t pos_out, u64 len, bool is_dedupe);
|
struct file *file_out, loff_t pos_out, u64 len, bool is_dedupe);
|
||||||
|
extern int xfs_reflink_inode_has_shared_extents(struct xfs_trans *tp,
|
||||||
|
struct xfs_inode *ip, bool *has_shared);
|
||||||
extern int xfs_reflink_clear_inode_flag(struct xfs_inode *ip,
|
extern int xfs_reflink_clear_inode_flag(struct xfs_inode *ip,
|
||||||
struct xfs_trans **tpp);
|
struct xfs_trans **tpp);
|
||||||
extern int xfs_reflink_unshare(struct xfs_inode *ip, xfs_off_t offset,
|
extern int xfs_reflink_unshare(struct xfs_inode *ip, xfs_off_t offset,
|
||||||
|
@ -1256,13 +1256,13 @@ xfs_rtpick_extent(
|
|||||||
{
|
{
|
||||||
xfs_rtblock_t b; /* result block */
|
xfs_rtblock_t b; /* result block */
|
||||||
int log2; /* log of sequence number */
|
int log2; /* log of sequence number */
|
||||||
__uint64_t resid; /* residual after log removed */
|
uint64_t resid; /* residual after log removed */
|
||||||
__uint64_t seq; /* sequence number of file creation */
|
uint64_t seq; /* sequence number of file creation */
|
||||||
__uint64_t *seqp; /* pointer to seqno in inode */
|
uint64_t *seqp; /* pointer to seqno in inode */
|
||||||
|
|
||||||
ASSERT(xfs_isilocked(mp->m_rbmip, XFS_ILOCK_EXCL));
|
ASSERT(xfs_isilocked(mp->m_rbmip, XFS_ILOCK_EXCL));
|
||||||
|
|
||||||
seqp = (__uint64_t *)&VFS_I(mp->m_rbmip)->i_atime;
|
seqp = (uint64_t *)&VFS_I(mp->m_rbmip)->i_atime;
|
||||||
if (!(mp->m_rbmip->i_d.di_flags & XFS_DIFLAG_NEWRTBM)) {
|
if (!(mp->m_rbmip->i_d.di_flags & XFS_DIFLAG_NEWRTBM)) {
|
||||||
mp->m_rbmip->i_d.di_flags |= XFS_DIFLAG_NEWRTBM;
|
mp->m_rbmip->i_d.di_flags |= XFS_DIFLAG_NEWRTBM;
|
||||||
*seqp = 0;
|
*seqp = 0;
|
||||||
|
@ -107,6 +107,8 @@ xfs_growfs_rt(
|
|||||||
/*
|
/*
|
||||||
* From xfs_rtbitmap.c
|
* From xfs_rtbitmap.c
|
||||||
*/
|
*/
|
||||||
|
int xfs_rtbuf_get(struct xfs_mount *mp, struct xfs_trans *tp,
|
||||||
|
xfs_rtblock_t block, int issum, struct xfs_buf **bpp);
|
||||||
int xfs_rtcheck_range(struct xfs_mount *mp, struct xfs_trans *tp,
|
int xfs_rtcheck_range(struct xfs_mount *mp, struct xfs_trans *tp,
|
||||||
xfs_rtblock_t start, xfs_extlen_t len, int val,
|
xfs_rtblock_t start, xfs_extlen_t len, int val,
|
||||||
xfs_rtblock_t *new, int *stat);
|
xfs_rtblock_t *new, int *stat);
|
||||||
@ -143,6 +145,7 @@ int xfs_rtalloc_query_all(struct xfs_trans *tp,
|
|||||||
# define xfs_growfs_rt(mp,in) (ENOSYS)
|
# define xfs_growfs_rt(mp,in) (ENOSYS)
|
||||||
# define xfs_rtalloc_query_range(t,l,h,f,p) (ENOSYS)
|
# define xfs_rtalloc_query_range(t,l,h,f,p) (ENOSYS)
|
||||||
# define xfs_rtalloc_query_all(t,f,p) (ENOSYS)
|
# define xfs_rtalloc_query_all(t,f,p) (ENOSYS)
|
||||||
|
# define xfs_rtbuf_get(m,t,b,i,p) (ENOSYS)
|
||||||
static inline int /* error */
|
static inline int /* error */
|
||||||
xfs_rtmount_init(
|
xfs_rtmount_init(
|
||||||
xfs_mount_t *mp) /* file system mount structure */
|
xfs_mount_t *mp) /* file system mount structure */
|
||||||
|
@ -33,9 +33,9 @@ int xfs_stats_format(struct xfsstats __percpu *stats, char *buf)
|
|||||||
{
|
{
|
||||||
int i, j;
|
int i, j;
|
||||||
int len = 0;
|
int len = 0;
|
||||||
__uint64_t xs_xstrat_bytes = 0;
|
uint64_t xs_xstrat_bytes = 0;
|
||||||
__uint64_t xs_write_bytes = 0;
|
uint64_t xs_write_bytes = 0;
|
||||||
__uint64_t xs_read_bytes = 0;
|
uint64_t xs_read_bytes = 0;
|
||||||
|
|
||||||
static const struct xstats_entry {
|
static const struct xstats_entry {
|
||||||
char *desc;
|
char *desc;
|
||||||
@ -100,7 +100,7 @@ int xfs_stats_format(struct xfsstats __percpu *stats, char *buf)
|
|||||||
void xfs_stats_clearall(struct xfsstats __percpu *stats)
|
void xfs_stats_clearall(struct xfsstats __percpu *stats)
|
||||||
{
|
{
|
||||||
int c;
|
int c;
|
||||||
__uint32_t vn_active;
|
uint32_t vn_active;
|
||||||
|
|
||||||
xfs_notice(NULL, "Clearing xfsstats");
|
xfs_notice(NULL, "Clearing xfsstats");
|
||||||
for_each_possible_cpu(c) {
|
for_each_possible_cpu(c) {
|
||||||
|
@ -54,125 +54,125 @@ enum {
|
|||||||
*/
|
*/
|
||||||
struct __xfsstats {
|
struct __xfsstats {
|
||||||
# define XFSSTAT_END_EXTENT_ALLOC 4
|
# define XFSSTAT_END_EXTENT_ALLOC 4
|
||||||
__uint32_t xs_allocx;
|
uint32_t xs_allocx;
|
||||||
__uint32_t xs_allocb;
|
uint32_t xs_allocb;
|
||||||
__uint32_t xs_freex;
|
uint32_t xs_freex;
|
||||||
__uint32_t xs_freeb;
|
uint32_t xs_freeb;
|
||||||
# define XFSSTAT_END_ALLOC_BTREE (XFSSTAT_END_EXTENT_ALLOC+4)
|
# define XFSSTAT_END_ALLOC_BTREE (XFSSTAT_END_EXTENT_ALLOC+4)
|
||||||
__uint32_t xs_abt_lookup;
|
uint32_t xs_abt_lookup;
|
||||||
__uint32_t xs_abt_compare;
|
uint32_t xs_abt_compare;
|
||||||
__uint32_t xs_abt_insrec;
|
uint32_t xs_abt_insrec;
|
||||||
__uint32_t xs_abt_delrec;
|
uint32_t xs_abt_delrec;
|
||||||
# define XFSSTAT_END_BLOCK_MAPPING (XFSSTAT_END_ALLOC_BTREE+7)
|
# define XFSSTAT_END_BLOCK_MAPPING (XFSSTAT_END_ALLOC_BTREE+7)
|
||||||
__uint32_t xs_blk_mapr;
|
uint32_t xs_blk_mapr;
|
||||||
__uint32_t xs_blk_mapw;
|
uint32_t xs_blk_mapw;
|
||||||
__uint32_t xs_blk_unmap;
|
uint32_t xs_blk_unmap;
|
||||||
__uint32_t xs_add_exlist;
|
uint32_t xs_add_exlist;
|
||||||
__uint32_t xs_del_exlist;
|
uint32_t xs_del_exlist;
|
||||||
__uint32_t xs_look_exlist;
|
uint32_t xs_look_exlist;
|
||||||
__uint32_t xs_cmp_exlist;
|
uint32_t xs_cmp_exlist;
|
||||||
# define XFSSTAT_END_BLOCK_MAP_BTREE (XFSSTAT_END_BLOCK_MAPPING+4)
|
# define XFSSTAT_END_BLOCK_MAP_BTREE (XFSSTAT_END_BLOCK_MAPPING+4)
|
||||||
__uint32_t xs_bmbt_lookup;
|
uint32_t xs_bmbt_lookup;
|
||||||
__uint32_t xs_bmbt_compare;
|
uint32_t xs_bmbt_compare;
|
||||||
__uint32_t xs_bmbt_insrec;
|
uint32_t xs_bmbt_insrec;
|
||||||
__uint32_t xs_bmbt_delrec;
|
uint32_t xs_bmbt_delrec;
|
||||||
# define XFSSTAT_END_DIRECTORY_OPS (XFSSTAT_END_BLOCK_MAP_BTREE+4)
|
# define XFSSTAT_END_DIRECTORY_OPS (XFSSTAT_END_BLOCK_MAP_BTREE+4)
|
||||||
__uint32_t xs_dir_lookup;
|
uint32_t xs_dir_lookup;
|
||||||
__uint32_t xs_dir_create;
|
uint32_t xs_dir_create;
|
||||||
__uint32_t xs_dir_remove;
|
uint32_t xs_dir_remove;
|
||||||
__uint32_t xs_dir_getdents;
|
uint32_t xs_dir_getdents;
|
||||||
# define XFSSTAT_END_TRANSACTIONS (XFSSTAT_END_DIRECTORY_OPS+3)
|
# define XFSSTAT_END_TRANSACTIONS (XFSSTAT_END_DIRECTORY_OPS+3)
|
||||||
__uint32_t xs_trans_sync;
|
uint32_t xs_trans_sync;
|
||||||
__uint32_t xs_trans_async;
|
uint32_t xs_trans_async;
|
||||||
__uint32_t xs_trans_empty;
|
uint32_t xs_trans_empty;
|
||||||
# define XFSSTAT_END_INODE_OPS (XFSSTAT_END_TRANSACTIONS+7)
|
# define XFSSTAT_END_INODE_OPS (XFSSTAT_END_TRANSACTIONS+7)
|
||||||
__uint32_t xs_ig_attempts;
|
uint32_t xs_ig_attempts;
|
||||||
__uint32_t xs_ig_found;
|
uint32_t xs_ig_found;
|
||||||
__uint32_t xs_ig_frecycle;
|
uint32_t xs_ig_frecycle;
|
||||||
__uint32_t xs_ig_missed;
|
uint32_t xs_ig_missed;
|
||||||
__uint32_t xs_ig_dup;
|
uint32_t xs_ig_dup;
|
||||||
__uint32_t xs_ig_reclaims;
|
uint32_t xs_ig_reclaims;
|
||||||
__uint32_t xs_ig_attrchg;
|
uint32_t xs_ig_attrchg;
|
||||||
# define XFSSTAT_END_LOG_OPS (XFSSTAT_END_INODE_OPS+5)
|
# define XFSSTAT_END_LOG_OPS (XFSSTAT_END_INODE_OPS+5)
|
||||||
__uint32_t xs_log_writes;
|
uint32_t xs_log_writes;
|
||||||
__uint32_t xs_log_blocks;
|
uint32_t xs_log_blocks;
|
||||||
__uint32_t xs_log_noiclogs;
|
uint32_t xs_log_noiclogs;
|
||||||
__uint32_t xs_log_force;
|
uint32_t xs_log_force;
|
||||||
__uint32_t xs_log_force_sleep;
|
uint32_t xs_log_force_sleep;
|
||||||
# define XFSSTAT_END_TAIL_PUSHING (XFSSTAT_END_LOG_OPS+10)
|
# define XFSSTAT_END_TAIL_PUSHING (XFSSTAT_END_LOG_OPS+10)
|
||||||
__uint32_t xs_try_logspace;
|
uint32_t xs_try_logspace;
|
||||||
__uint32_t xs_sleep_logspace;
|
uint32_t xs_sleep_logspace;
|
||||||
__uint32_t xs_push_ail;
|
uint32_t xs_push_ail;
|
||||||
__uint32_t xs_push_ail_success;
|
uint32_t xs_push_ail_success;
|
||||||
__uint32_t xs_push_ail_pushbuf;
|
uint32_t xs_push_ail_pushbuf;
|
||||||
__uint32_t xs_push_ail_pinned;
|
uint32_t xs_push_ail_pinned;
|
||||||
__uint32_t xs_push_ail_locked;
|
uint32_t xs_push_ail_locked;
|
||||||
__uint32_t xs_push_ail_flushing;
|
uint32_t xs_push_ail_flushing;
|
||||||
__uint32_t xs_push_ail_restarts;
|
uint32_t xs_push_ail_restarts;
|
||||||
__uint32_t xs_push_ail_flush;
|
uint32_t xs_push_ail_flush;
|
||||||
# define XFSSTAT_END_WRITE_CONVERT (XFSSTAT_END_TAIL_PUSHING+2)
|
# define XFSSTAT_END_WRITE_CONVERT (XFSSTAT_END_TAIL_PUSHING+2)
|
||||||
__uint32_t xs_xstrat_quick;
|
uint32_t xs_xstrat_quick;
|
||||||
__uint32_t xs_xstrat_split;
|
uint32_t xs_xstrat_split;
|
||||||
# define XFSSTAT_END_READ_WRITE_OPS (XFSSTAT_END_WRITE_CONVERT+2)
|
# define XFSSTAT_END_READ_WRITE_OPS (XFSSTAT_END_WRITE_CONVERT+2)
|
||||||
__uint32_t xs_write_calls;
|
uint32_t xs_write_calls;
|
||||||
__uint32_t xs_read_calls;
|
uint32_t xs_read_calls;
|
||||||
# define XFSSTAT_END_ATTRIBUTE_OPS (XFSSTAT_END_READ_WRITE_OPS+4)
|
# define XFSSTAT_END_ATTRIBUTE_OPS (XFSSTAT_END_READ_WRITE_OPS+4)
|
||||||
__uint32_t xs_attr_get;
|
uint32_t xs_attr_get;
|
||||||
__uint32_t xs_attr_set;
|
uint32_t xs_attr_set;
|
||||||
__uint32_t xs_attr_remove;
|
uint32_t xs_attr_remove;
|
||||||
__uint32_t xs_attr_list;
|
uint32_t xs_attr_list;
|
||||||
# define XFSSTAT_END_INODE_CLUSTER (XFSSTAT_END_ATTRIBUTE_OPS+3)
|
# define XFSSTAT_END_INODE_CLUSTER (XFSSTAT_END_ATTRIBUTE_OPS+3)
|
||||||
__uint32_t xs_iflush_count;
|
uint32_t xs_iflush_count;
|
||||||
__uint32_t xs_icluster_flushcnt;
|
uint32_t xs_icluster_flushcnt;
|
||||||
__uint32_t xs_icluster_flushinode;
|
uint32_t xs_icluster_flushinode;
|
||||||
# define XFSSTAT_END_VNODE_OPS (XFSSTAT_END_INODE_CLUSTER+8)
|
# define XFSSTAT_END_VNODE_OPS (XFSSTAT_END_INODE_CLUSTER+8)
|
||||||
__uint32_t vn_active; /* # vnodes not on free lists */
|
uint32_t vn_active; /* # vnodes not on free lists */
|
||||||
__uint32_t vn_alloc; /* # times vn_alloc called */
|
uint32_t vn_alloc; /* # times vn_alloc called */
|
||||||
__uint32_t vn_get; /* # times vn_get called */
|
uint32_t vn_get; /* # times vn_get called */
|
||||||
__uint32_t vn_hold; /* # times vn_hold called */
|
uint32_t vn_hold; /* # times vn_hold called */
|
||||||
__uint32_t vn_rele; /* # times vn_rele called */
|
uint32_t vn_rele; /* # times vn_rele called */
|
||||||
__uint32_t vn_reclaim; /* # times vn_reclaim called */
|
uint32_t vn_reclaim; /* # times vn_reclaim called */
|
||||||
__uint32_t vn_remove; /* # times vn_remove called */
|
uint32_t vn_remove; /* # times vn_remove called */
|
||||||
__uint32_t vn_free; /* # times vn_free called */
|
uint32_t vn_free; /* # times vn_free called */
|
||||||
#define XFSSTAT_END_BUF (XFSSTAT_END_VNODE_OPS+9)
|
#define XFSSTAT_END_BUF (XFSSTAT_END_VNODE_OPS+9)
|
||||||
__uint32_t xb_get;
|
uint32_t xb_get;
|
||||||
__uint32_t xb_create;
|
uint32_t xb_create;
|
||||||
__uint32_t xb_get_locked;
|
uint32_t xb_get_locked;
|
||||||
__uint32_t xb_get_locked_waited;
|
uint32_t xb_get_locked_waited;
|
||||||
__uint32_t xb_busy_locked;
|
uint32_t xb_busy_locked;
|
||||||
__uint32_t xb_miss_locked;
|
uint32_t xb_miss_locked;
|
||||||
__uint32_t xb_page_retries;
|
uint32_t xb_page_retries;
|
||||||
__uint32_t xb_page_found;
|
uint32_t xb_page_found;
|
||||||
__uint32_t xb_get_read;
|
uint32_t xb_get_read;
|
||||||
/* Version 2 btree counters */
|
/* Version 2 btree counters */
|
||||||
#define XFSSTAT_END_ABTB_V2 (XFSSTAT_END_BUF + __XBTS_MAX)
|
#define XFSSTAT_END_ABTB_V2 (XFSSTAT_END_BUF + __XBTS_MAX)
|
||||||
__uint32_t xs_abtb_2[__XBTS_MAX];
|
uint32_t xs_abtb_2[__XBTS_MAX];
|
||||||
#define XFSSTAT_END_ABTC_V2 (XFSSTAT_END_ABTB_V2 + __XBTS_MAX)
|
#define XFSSTAT_END_ABTC_V2 (XFSSTAT_END_ABTB_V2 + __XBTS_MAX)
|
||||||
__uint32_t xs_abtc_2[__XBTS_MAX];
|
uint32_t xs_abtc_2[__XBTS_MAX];
|
||||||
#define XFSSTAT_END_BMBT_V2 (XFSSTAT_END_ABTC_V2 + __XBTS_MAX)
|
#define XFSSTAT_END_BMBT_V2 (XFSSTAT_END_ABTC_V2 + __XBTS_MAX)
|
||||||
__uint32_t xs_bmbt_2[__XBTS_MAX];
|
uint32_t xs_bmbt_2[__XBTS_MAX];
|
||||||
#define XFSSTAT_END_IBT_V2 (XFSSTAT_END_BMBT_V2 + __XBTS_MAX)
|
#define XFSSTAT_END_IBT_V2 (XFSSTAT_END_BMBT_V2 + __XBTS_MAX)
|
||||||
__uint32_t xs_ibt_2[__XBTS_MAX];
|
uint32_t xs_ibt_2[__XBTS_MAX];
|
||||||
#define XFSSTAT_END_FIBT_V2 (XFSSTAT_END_IBT_V2 + __XBTS_MAX)
|
#define XFSSTAT_END_FIBT_V2 (XFSSTAT_END_IBT_V2 + __XBTS_MAX)
|
||||||
__uint32_t xs_fibt_2[__XBTS_MAX];
|
uint32_t xs_fibt_2[__XBTS_MAX];
|
||||||
#define XFSSTAT_END_RMAP_V2 (XFSSTAT_END_FIBT_V2 + __XBTS_MAX)
|
#define XFSSTAT_END_RMAP_V2 (XFSSTAT_END_FIBT_V2 + __XBTS_MAX)
|
||||||
__uint32_t xs_rmap_2[__XBTS_MAX];
|
uint32_t xs_rmap_2[__XBTS_MAX];
|
||||||
#define XFSSTAT_END_REFCOUNT (XFSSTAT_END_RMAP_V2 + __XBTS_MAX)
|
#define XFSSTAT_END_REFCOUNT (XFSSTAT_END_RMAP_V2 + __XBTS_MAX)
|
||||||
__uint32_t xs_refcbt_2[__XBTS_MAX];
|
uint32_t xs_refcbt_2[__XBTS_MAX];
|
||||||
#define XFSSTAT_END_XQMSTAT (XFSSTAT_END_REFCOUNT + 6)
|
#define XFSSTAT_END_XQMSTAT (XFSSTAT_END_REFCOUNT + 6)
|
||||||
__uint32_t xs_qm_dqreclaims;
|
uint32_t xs_qm_dqreclaims;
|
||||||
__uint32_t xs_qm_dqreclaim_misses;
|
uint32_t xs_qm_dqreclaim_misses;
|
||||||
__uint32_t xs_qm_dquot_dups;
|
uint32_t xs_qm_dquot_dups;
|
||||||
__uint32_t xs_qm_dqcachemisses;
|
uint32_t xs_qm_dqcachemisses;
|
||||||
__uint32_t xs_qm_dqcachehits;
|
uint32_t xs_qm_dqcachehits;
|
||||||
__uint32_t xs_qm_dqwants;
|
uint32_t xs_qm_dqwants;
|
||||||
#define XFSSTAT_END_QM (XFSSTAT_END_XQMSTAT+2)
|
#define XFSSTAT_END_QM (XFSSTAT_END_XQMSTAT+2)
|
||||||
__uint32_t xs_qm_dquot;
|
uint32_t xs_qm_dquot;
|
||||||
__uint32_t xs_qm_dquot_unused;
|
uint32_t xs_qm_dquot_unused;
|
||||||
/* Extra precision counters */
|
/* Extra precision counters */
|
||||||
__uint64_t xs_xstrat_bytes;
|
uint64_t xs_xstrat_bytes;
|
||||||
__uint64_t xs_write_bytes;
|
uint64_t xs_write_bytes;
|
||||||
__uint64_t xs_read_bytes;
|
uint64_t xs_read_bytes;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct xfsstats {
|
struct xfsstats {
|
||||||
@ -186,7 +186,7 @@ struct xfsstats {
|
|||||||
* simple wrapper for getting the array index of s struct member offset
|
* simple wrapper for getting the array index of s struct member offset
|
||||||
*/
|
*/
|
||||||
#define XFS_STATS_CALC_INDEX(member) \
|
#define XFS_STATS_CALC_INDEX(member) \
|
||||||
(offsetof(struct __xfsstats, member) / (int)sizeof(__uint32_t))
|
(offsetof(struct __xfsstats, member) / (int)sizeof(uint32_t))
|
||||||
|
|
||||||
|
|
||||||
int xfs_stats_format(struct xfsstats __percpu *stats, char *buf);
|
int xfs_stats_format(struct xfsstats __percpu *stats, char *buf);
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user