mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-12 08:09:56 +00:00
Merge git://oss.sgi.com:8090/xfs/linux-2.6
* git://oss.sgi.com:8090/xfs/linux-2.6: [XFS] Remove xfs_iext_irec_compact_full() [XFS] Fix extent list corruption in xfs_iext_irec_compact_full().
This commit is contained in:
commit
ec4d90287e
@ -4118,7 +4118,7 @@ xfs_iext_indirect_to_direct(
|
||||
ASSERT(nextents <= XFS_LINEAR_EXTS);
|
||||
size = nextents * sizeof(xfs_bmbt_rec_t);
|
||||
|
||||
xfs_iext_irec_compact_full(ifp);
|
||||
xfs_iext_irec_compact_pages(ifp);
|
||||
ASSERT(ifp->if_real_bytes == XFS_IEXT_BUFSZ);
|
||||
|
||||
ep = ifp->if_u1.if_ext_irec->er_extbuf;
|
||||
@ -4449,8 +4449,7 @@ xfs_iext_irec_remove(
|
||||
* compaction policy is as follows:
|
||||
*
|
||||
* Full Compaction: Extents fit into a single page (or inline buffer)
|
||||
* Full Compaction: Extents occupy less than 10% of allocated space
|
||||
* Partial Compaction: Extents occupy > 10% and < 50% of allocated space
|
||||
* Partial Compaction: Extents occupy less than 50% of allocated space
|
||||
* No Compaction: Extents occupy at least 50% of allocated space
|
||||
*/
|
||||
void
|
||||
@ -4471,8 +4470,6 @@ xfs_iext_irec_compact(
|
||||
xfs_iext_direct_to_inline(ifp, nextents);
|
||||
} else if (nextents <= XFS_LINEAR_EXTS) {
|
||||
xfs_iext_indirect_to_direct(ifp);
|
||||
} else if (nextents < (nlists * XFS_LINEAR_EXTS) >> 3) {
|
||||
xfs_iext_irec_compact_full(ifp);
|
||||
} else if (nextents < (nlists * XFS_LINEAR_EXTS) >> 1) {
|
||||
xfs_iext_irec_compact_pages(ifp);
|
||||
}
|
||||
@ -4496,7 +4493,7 @@ xfs_iext_irec_compact_pages(
|
||||
erp_next = erp + 1;
|
||||
if (erp_next->er_extcount <=
|
||||
(XFS_LINEAR_EXTS - erp->er_extcount)) {
|
||||
memmove(&erp->er_extbuf[erp->er_extcount],
|
||||
memcpy(&erp->er_extbuf[erp->er_extcount],
|
||||
erp_next->er_extbuf, erp_next->er_extcount *
|
||||
sizeof(xfs_bmbt_rec_t));
|
||||
erp->er_extcount += erp_next->er_extcount;
|
||||
@ -4515,91 +4512,6 @@ xfs_iext_irec_compact_pages(
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Fully compact the extent records managed by the indirection array.
|
||||
*/
|
||||
void
|
||||
xfs_iext_irec_compact_full(
|
||||
xfs_ifork_t *ifp) /* inode fork pointer */
|
||||
{
|
||||
xfs_bmbt_rec_host_t *ep, *ep_next; /* extent record pointers */
|
||||
xfs_ext_irec_t *erp, *erp_next; /* extent irec pointers */
|
||||
int erp_idx = 0; /* extent irec index */
|
||||
int ext_avail; /* empty entries in ex list */
|
||||
int ext_diff; /* number of exts to add */
|
||||
int nlists; /* number of irec's (ex lists) */
|
||||
|
||||
ASSERT(ifp->if_flags & XFS_IFEXTIREC);
|
||||
|
||||
nlists = ifp->if_real_bytes / XFS_IEXT_BUFSZ;
|
||||
erp = ifp->if_u1.if_ext_irec;
|
||||
ep = &erp->er_extbuf[erp->er_extcount];
|
||||
erp_next = erp + 1;
|
||||
ep_next = erp_next->er_extbuf;
|
||||
|
||||
while (erp_idx < nlists - 1) {
|
||||
/*
|
||||
* Check how many extent records are available in this irec.
|
||||
* If there is none skip the whole exercise.
|
||||
*/
|
||||
ext_avail = XFS_LINEAR_EXTS - erp->er_extcount;
|
||||
if (ext_avail) {
|
||||
|
||||
/*
|
||||
* Copy over as many as possible extent records into
|
||||
* the previous page.
|
||||
*/
|
||||
ext_diff = MIN(ext_avail, erp_next->er_extcount);
|
||||
memcpy(ep, ep_next, ext_diff * sizeof(xfs_bmbt_rec_t));
|
||||
erp->er_extcount += ext_diff;
|
||||
erp_next->er_extcount -= ext_diff;
|
||||
|
||||
/*
|
||||
* If the next irec is empty now we can simply
|
||||
* remove it.
|
||||
*/
|
||||
if (erp_next->er_extcount == 0) {
|
||||
/*
|
||||
* Free page before removing extent record
|
||||
* so er_extoffs don't get modified in
|
||||
* xfs_iext_irec_remove.
|
||||
*/
|
||||
kmem_free(erp_next->er_extbuf);
|
||||
erp_next->er_extbuf = NULL;
|
||||
xfs_iext_irec_remove(ifp, erp_idx + 1);
|
||||
erp = &ifp->if_u1.if_ext_irec[erp_idx];
|
||||
nlists = ifp->if_real_bytes / XFS_IEXT_BUFSZ;
|
||||
|
||||
/*
|
||||
* If the next irec is not empty move up the content
|
||||
* that has not been copied to the previous page to
|
||||
* the beggining of this one.
|
||||
*/
|
||||
} else {
|
||||
memmove(erp_next->er_extbuf, &ep_next[ext_diff],
|
||||
erp_next->er_extcount *
|
||||
sizeof(xfs_bmbt_rec_t));
|
||||
ep_next = erp_next->er_extbuf;
|
||||
memset(&ep_next[erp_next->er_extcount], 0,
|
||||
(XFS_LINEAR_EXTS -
|
||||
erp_next->er_extcount) *
|
||||
sizeof(xfs_bmbt_rec_t));
|
||||
}
|
||||
}
|
||||
|
||||
if (erp->er_extcount == XFS_LINEAR_EXTS) {
|
||||
erp_idx++;
|
||||
if (erp_idx < nlists)
|
||||
erp = &ifp->if_u1.if_ext_irec[erp_idx];
|
||||
else
|
||||
break;
|
||||
}
|
||||
ep = &erp->er_extbuf[erp->er_extcount];
|
||||
erp_next = erp + 1;
|
||||
ep_next = erp_next->er_extbuf;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* This is called to update the er_extoff field in the indirection
|
||||
* array when extents have been added or removed from one of the
|
||||
|
Loading…
x
Reference in New Issue
Block a user