Commit 5ae1a1f8 authored by Linus Torvalds's avatar Linus Torvalds

Merge bk://kernel.bkbits.net/lord/xfs-2.5

into home.osdl.org:/home/torvalds/v2.5/linux
parents f793f34c 130b6627
...@@ -803,7 +803,7 @@ page_state_convert( ...@@ -803,7 +803,7 @@ page_state_convert(
bh = bh->b_this_page; bh = bh->b_this_page;
} while (offset < end_offset); } while (offset < end_offset);
if (uptodate) if (uptodate && bh == head)
SetPageUptodate(page); SetPageUptodate(page);
if (startio) if (startio)
......
...@@ -262,7 +262,11 @@ linvfs_readdir( ...@@ -262,7 +262,11 @@ linvfs_readdir(
uio.uio_iov = &iov; uio.uio_iov = &iov;
uio.uio_fmode = filp->f_mode; uio.uio_fmode = filp->f_mode;
uio.uio_segflg = UIO_SYSSPACE; uio.uio_segflg = UIO_SYSSPACE;
curr_offset = uio.uio_offset = filp->f_pos; curr_offset = filp->f_pos;
if (filp->f_pos != 0x7fffffff)
uio.uio_offset = filp->f_pos;
else
uio.uio_offset = 0xffffffff;
while (!eof) { while (!eof) {
uio.uio_resid = iov.iov_len = rlen; uio.uio_resid = iov.iov_len = rlen;
...@@ -283,13 +287,13 @@ linvfs_readdir( ...@@ -283,13 +287,13 @@ linvfs_readdir(
namelen = strlen(dbp->d_name); namelen = strlen(dbp->d_name);
if (filldir(dirent, dbp->d_name, namelen, if (filldir(dirent, dbp->d_name, namelen,
(loff_t) curr_offset, (loff_t) curr_offset & 0x7fffffff,
(ino_t) dbp->d_ino, (ino_t) dbp->d_ino,
DT_UNKNOWN)) { DT_UNKNOWN)) {
goto done; goto done;
} }
size -= dbp->d_reclen; size -= dbp->d_reclen;
curr_offset = (loff_t)dbp->d_off & 0x7fffffff; curr_offset = (loff_t)dbp->d_off /* & 0x7fffffff */;
dbp = nextdp(dbp); dbp = nextdp(dbp);
} }
} }
......
...@@ -109,14 +109,20 @@ linvfs_mknod( ...@@ -109,14 +109,20 @@ linvfs_mknod(
struct inode *ip; struct inode *ip;
vattr_t va; vattr_t va;
vnode_t *vp = NULL, *dvp = LINVFS_GET_VP(dir); vnode_t *vp = NULL, *dvp = LINVFS_GET_VP(dir);
xfs_acl_t *default_acl = NULL;
xattr_exists_t test_default_acl = _ACL_DEFAULT_EXISTS; xattr_exists_t test_default_acl = _ACL_DEFAULT_EXISTS;
int have_default_acl = 0; int error;
int error = EINVAL;
if (test_default_acl) if (test_default_acl && test_default_acl(dvp)) {
have_default_acl = test_default_acl(dvp); if (!_ACL_ALLOC(default_acl))
return -ENOMEM;
if (!_ACL_GET_DEFAULT(dvp, default_acl)) {
_ACL_FREE(default_acl);
default_acl = NULL;
}
}
if (IS_POSIXACL(dir) && !have_default_acl && has_fs_struct(current)) if (IS_POSIXACL(dir) && !default_acl && has_fs_struct(current))
mode &= ~current->fs->umask; mode &= ~current->fs->umask;
memset(&va, 0, sizeof(va)); memset(&va, 0, sizeof(va));
...@@ -140,13 +146,36 @@ linvfs_mknod( ...@@ -140,13 +146,36 @@ linvfs_mknod(
break; break;
} }
if (default_acl) {
if (!error) { if (!error) {
ASSERT(vp); error = _ACL_INHERIT(vp, &va, default_acl);
ip = LINVFS_GET_IP(vp); if (!error) {
if (!ip) { VMODIFY(vp);
} else {
struct dentry teardown = {};
int err2;
/* Oh, the horror.
* If we can't add the ACL we must back out.
* ENOSPC can hit here, among other things.
*/
teardown.d_inode = ip = LINVFS_GET_IP(vp);
teardown.d_name = dentry->d_name;
remove_inode_hash(ip);
make_bad_inode(ip);
if (S_ISDIR(mode))
VOP_RMDIR(dvp, &teardown, NULL, err2);
else
VOP_REMOVE(dvp, &teardown, NULL, err2);
VN_RELE(vp); VN_RELE(vp);
return -ENOMEM;
} }
}
_ACL_FREE(default_acl);
}
if (!error) {
ASSERT(vp);
ip = LINVFS_GET_IP(vp);
if (S_ISCHR(mode) || S_ISBLK(mode)) if (S_ISCHR(mode) || S_ISBLK(mode))
ip->i_rdev = to_kdev_t(rdev); ip->i_rdev = to_kdev_t(rdev);
...@@ -155,19 +184,6 @@ linvfs_mknod( ...@@ -155,19 +184,6 @@ linvfs_mknod(
d_instantiate(dentry, ip); d_instantiate(dentry, ip);
validate_fields(dir); validate_fields(dir);
} }
if (!error && have_default_acl) {
_ACL_DECL (pdacl);
if (!_ACL_ALLOC(pdacl)) {
error = -ENOMEM;
} else {
if (_ACL_GET_DEFAULT(dvp, pdacl))
error = _ACL_INHERIT(vp, &va, pdacl);
VMODIFY(vp);
_ACL_FREE(pdacl);
}
}
return -error; return -error;
} }
......
...@@ -506,6 +506,24 @@ linvfs_write_super( ...@@ -506,6 +506,24 @@ linvfs_write_super(
sb->s_dirt = 0; sb->s_dirt = 0;
} }
STATIC int
linvfs_sync_super(
struct super_block *sb,
int wait)
{
vfs_t *vfsp = LINVFS_GET_VFS(sb);
int error;
int flags = SYNC_FSDATA;
if (wait)
flags |= SYNC_WAIT;
VFS_SYNC(vfsp, flags, NULL, error);
sb->s_dirt = 0;
return -error;
}
STATIC int STATIC int
linvfs_statfs( linvfs_statfs(
struct super_block *sb, struct super_block *sb,
...@@ -798,6 +816,7 @@ STATIC struct super_operations linvfs_sops = { ...@@ -798,6 +816,7 @@ STATIC struct super_operations linvfs_sops = {
.clear_inode = linvfs_clear_inode, .clear_inode = linvfs_clear_inode,
.put_super = linvfs_put_super, .put_super = linvfs_put_super,
.write_super = linvfs_write_super, .write_super = linvfs_write_super,
.sync_fs = linvfs_sync_super,
.write_super_lockfs = linvfs_freeze_fs, .write_super_lockfs = linvfs_freeze_fs,
.unlockfs = linvfs_unfreeze_fs, .unlockfs = linvfs_unfreeze_fs,
.statfs = linvfs_statfs, .statfs = linvfs_statfs,
......
...@@ -1114,10 +1114,10 @@ _pagebuf_wait_unpin( ...@@ -1114,10 +1114,10 @@ _pagebuf_wait_unpin(
add_wait_queue(&pb->pb_waiters, &wait); add_wait_queue(&pb->pb_waiters, &wait);
for (;;) { for (;;) {
current->state = TASK_UNINTERRUPTIBLE; current->state = TASK_UNINTERRUPTIBLE;
if (atomic_read(&pb->pb_pin_count) == 0) { if (atomic_read(&pb->pb_pin_count) == 0)
break; break;
} if (atomic_read(&pb->pb_io_remaining))
pagebuf_run_queues(pb); blk_run_queues();
schedule(); schedule();
} }
remove_wait_queue(&pb->pb_waiters, &wait); remove_wait_queue(&pb->pb_waiters, &wait);
...@@ -1224,26 +1224,27 @@ pagebuf_iostart( /* start I/O on a buffer */ ...@@ -1224,26 +1224,27 @@ pagebuf_iostart( /* start I/O on a buffer */
return status; return status;
} }
pb->pb_flags &= pb->pb_flags &= ~(PBF_READ | PBF_WRITE | PBF_ASYNC | \
~(PBF_READ|PBF_WRITE|PBF_ASYNC|PBF_DELWRI|PBF_READ_AHEAD); PBF_DELWRI | PBF_READ_AHEAD | PBF_RUN_QUEUES);
pb->pb_flags |= flags & pb->pb_flags |= flags & (PBF_READ | PBF_WRITE | PBF_ASYNC | \
(PBF_READ|PBF_WRITE|PBF_ASYNC|PBF_SYNC|PBF_READ_AHEAD); PBF_SYNC | PBF_READ_AHEAD | PBF_RUN_QUEUES);
BUG_ON(pb->pb_bn == PAGE_BUF_DADDR_NULL); BUG_ON(pb->pb_bn == PAGE_BUF_DADDR_NULL);
/* For writes call internal function which checks for /* For writes allow an alternate strategy routine to precede
* filesystem specific callout function and execute it. * the actual I/O request (which may not be issued at all in
* a shutdown situation, for example).
*/ */
if (flags & PBF_WRITE) { status = (flags & PBF_WRITE) ?
status = __pagebuf_iorequest(pb); pagebuf_iostrategy(pb) : pagebuf_iorequest(pb);
} else {
status = pagebuf_iorequest(pb);
}
/* Wait for I/O if we are not an async request */ /* Wait for I/O if we are not an async request.
if ((status == 0) && (flags & PBF_ASYNC) == 0) { * Note: async I/O request completion will release the buffer,
* and that can already be done by this point. So using the
* buffer pointer from here on, after async I/O, is invalid.
*/
if (!status && !(flags & PBF_ASYNC))
status = pagebuf_iowait(pb); status = pagebuf_iowait(pb);
}
return status; return status;
} }
...@@ -1381,8 +1382,6 @@ _pagebuf_ioapply( ...@@ -1381,8 +1382,6 @@ _pagebuf_ioapply(
nr_pages = total_nr_pages; nr_pages = total_nr_pages;
bio = bio_alloc(GFP_NOIO, nr_pages); bio = bio_alloc(GFP_NOIO, nr_pages);
BUG_ON(bio == NULL);
bio->bi_bdev = pb->pb_target->pbr_bdev; bio->bi_bdev = pb->pb_target->pbr_bdev;
bio->bi_sector = sector; bio->bi_sector = sector;
bio->bi_end_io = bio_end_io_pagebuf; bio->bi_end_io = bio_end_io_pagebuf;
...@@ -1418,6 +1417,12 @@ _pagebuf_ioapply( ...@@ -1418,6 +1417,12 @@ _pagebuf_ioapply(
} else { } else {
pagebuf_ioerror(pb, EIO); pagebuf_ioerror(pb, EIO);
} }
if (pb->pb_flags & PBF_RUN_QUEUES) {
pb->pb_flags &= ~PBF_RUN_QUEUES;
if (atomic_read(&pb->pb_io_remaining) > 1)
blk_run_queues();
}
} }
/* /*
...@@ -1453,6 +1458,8 @@ pagebuf_iorequest( /* start real I/O */ ...@@ -1453,6 +1458,8 @@ pagebuf_iorequest( /* start real I/O */
_pagebuf_wait_unpin(pb); _pagebuf_wait_unpin(pb);
} }
pagebuf_hold(pb);
/* Set the count to 1 initially, this will stop an I/O /* Set the count to 1 initially, this will stop an I/O
* completion callout which happens before we have started * completion callout which happens before we have started
* all the I/O from calling pagebuf_iodone too early. * all the I/O from calling pagebuf_iodone too early.
...@@ -1460,6 +1467,8 @@ pagebuf_iorequest( /* start real I/O */ ...@@ -1460,6 +1467,8 @@ pagebuf_iorequest( /* start real I/O */
atomic_set(&pb->pb_io_remaining, 1); atomic_set(&pb->pb_io_remaining, 1);
_pagebuf_ioapply(pb); _pagebuf_ioapply(pb);
_pagebuf_iodone(pb, 0); _pagebuf_iodone(pb, 0);
pagebuf_rele(pb);
return 0; return 0;
} }
...@@ -1475,7 +1484,8 @@ pagebuf_iowait( ...@@ -1475,7 +1484,8 @@ pagebuf_iowait(
page_buf_t *pb) page_buf_t *pb)
{ {
PB_TRACE(pb, PB_TRACE_REC(iowait), 0); PB_TRACE(pb, PB_TRACE_REC(iowait), 0);
pagebuf_run_queues(pb); if (atomic_read(&pb->pb_io_remaining))
blk_run_queues();
down(&pb->pb_iodonesema); down(&pb->pb_iodonesema);
PB_TRACE(pb, PB_TRACE_REC(iowaited), (int)pb->pb_error); PB_TRACE(pb, PB_TRACE_REC(iowaited), (int)pb->pb_error);
return pb->pb_error; return pb->pb_error;
...@@ -1554,6 +1564,7 @@ pagebuf_iomove( ...@@ -1554,6 +1564,7 @@ pagebuf_iomove(
} }
} }
/* /*
* Pagebuf delayed write buffer handling * Pagebuf delayed write buffer handling
*/ */
...@@ -1683,13 +1694,13 @@ pagebuf_daemon( ...@@ -1683,13 +1694,13 @@ pagebuf_daemon(
pb->pb_flags &= ~PBF_DELWRI; pb->pb_flags &= ~PBF_DELWRI;
pb->pb_flags |= PBF_WRITE; pb->pb_flags |= PBF_WRITE;
__pagebuf_iorequest(pb); pagebuf_iostrategy(pb);
} }
if (as_list_len > 0) if (as_list_len > 0)
purge_addresses(); purge_addresses();
if (count) if (count)
pagebuf_run_queues(NULL); blk_run_queues();
force_flush = 0; force_flush = 0;
} while (pbd_active == 1); } while (pbd_active == 1);
...@@ -1756,9 +1767,9 @@ pagebuf_delwri_flush( ...@@ -1756,9 +1767,9 @@ pagebuf_delwri_flush(
pb->pb_flags &= ~PBF_DELWRI; pb->pb_flags &= ~PBF_DELWRI;
pb->pb_flags |= PBF_WRITE; pb->pb_flags |= PBF_WRITE;
__pagebuf_iorequest(pb); pagebuf_iostrategy(pb);
if (++flush_cnt > 32) { if (++flush_cnt > 32) {
pagebuf_run_queues(NULL); blk_run_queues();
flush_cnt = 0; flush_cnt = 0;
} }
...@@ -1767,7 +1778,7 @@ pagebuf_delwri_flush( ...@@ -1767,7 +1778,7 @@ pagebuf_delwri_flush(
spin_unlock(&pbd_delwrite_lock); spin_unlock(&pbd_delwrite_lock);
pagebuf_run_queues(NULL); blk_run_queues();
if (pinptr) if (pinptr)
*pinptr = pincount; *pinptr = pincount;
......
...@@ -128,6 +128,7 @@ typedef enum page_buf_flags_e { /* pb_flags values */ ...@@ -128,6 +128,7 @@ typedef enum page_buf_flags_e { /* pb_flags values */
PBF_FORCEIO = (1 << 21), PBF_FORCEIO = (1 << 21),
PBF_FLUSH = (1 << 22), /* flush disk write cache */ PBF_FLUSH = (1 << 22), /* flush disk write cache */
PBF_READ_AHEAD = (1 << 23), PBF_READ_AHEAD = (1 << 23),
PBF_RUN_QUEUES = (1 << 24), /* run block device task queue */
} page_buf_flags_t; } page_buf_flags_t;
...@@ -239,10 +240,6 @@ typedef struct page_buf_s { ...@@ -239,10 +240,6 @@ typedef struct page_buf_s {
} page_buf_t; } page_buf_t;
/*
* page_buf module entry points
*/
/* Finding and Reading Buffers */ /* Finding and Reading Buffers */
extern page_buf_t *pagebuf_find( /* find buffer for block if */ extern page_buf_t *pagebuf_find( /* find buffer for block if */
...@@ -281,7 +278,6 @@ extern int pagebuf_associate_memory( ...@@ -281,7 +278,6 @@ extern int pagebuf_associate_memory(
void *, void *,
size_t); size_t);
extern void pagebuf_hold( /* increment reference count */ extern void pagebuf_hold( /* increment reference count */
page_buf_t *); /* buffer to hold */ page_buf_t *); /* buffer to hold */
...@@ -291,7 +287,7 @@ extern void pagebuf_readahead( /* read ahead into cache */ ...@@ -291,7 +287,7 @@ extern void pagebuf_readahead( /* read ahead into cache */
size_t, /* length of range */ size_t, /* length of range */
page_buf_flags_t); /* additional read flags */ page_buf_flags_t); /* additional read flags */
/* Writing and Releasing Buffers */ /* Releasing Buffers */
extern void pagebuf_free( /* deallocate a buffer */ extern void pagebuf_free( /* deallocate a buffer */
page_buf_t *); /* buffer to deallocate */ page_buf_t *); /* buffer to deallocate */
...@@ -314,11 +310,7 @@ extern int pagebuf_lock( /* lock buffer */ ...@@ -314,11 +310,7 @@ extern int pagebuf_lock( /* lock buffer */
extern void pagebuf_unlock( /* unlock buffer */ extern void pagebuf_unlock( /* unlock buffer */
page_buf_t *); /* buffer to unlock */ page_buf_t *); /* buffer to unlock */
/* Buffer Utility Routines */ /* Buffer Read and Write Routines */
static inline int pagebuf_geterror(page_buf_t *pb)
{
return (pb ? pb->pb_error : ENOMEM);
}
extern void pagebuf_iodone( /* mark buffer I/O complete */ extern void pagebuf_iodone( /* mark buffer I/O complete */
page_buf_t *, /* buffer to mark */ page_buf_t *, /* buffer to mark */
...@@ -339,21 +331,9 @@ extern int pagebuf_iostart( /* start I/O on a buffer */ ...@@ -339,21 +331,9 @@ extern int pagebuf_iostart( /* start I/O on a buffer */
extern int pagebuf_iorequest( /* start real I/O */ extern int pagebuf_iorequest( /* start real I/O */
page_buf_t *); /* buffer to convey to device */ page_buf_t *); /* buffer to convey to device */
/*
* pagebuf_iorequest is the core I/O request routine.
* It assumes that the buffer is well-formed and
* mapped and ready for physical I/O, unlike
* pagebuf_iostart() and pagebuf_iophysio(). Those
* routines call the inode pagebuf_ioinitiate routine to start I/O,
* if it is present, or else call pagebuf_iorequest()
* directly if the inode pagebuf_ioinitiate routine is not present.
*/
extern int pagebuf_iowait( /* wait for buffer I/O done */ extern int pagebuf_iowait( /* wait for buffer I/O done */
page_buf_t *); /* buffer to wait on */ page_buf_t *); /* buffer to wait on */
extern caddr_t pagebuf_offset(page_buf_t *, size_t);
extern void pagebuf_iomove( /* move data in/out of pagebuf */ extern void pagebuf_iomove( /* move data in/out of pagebuf */
page_buf_t *, /* buffer to manipulate */ page_buf_t *, /* buffer to manipulate */
size_t, /* starting buffer offset */ size_t, /* starting buffer offset */
...@@ -361,6 +341,22 @@ extern void pagebuf_iomove( /* move data in/out of pagebuf */ ...@@ -361,6 +341,22 @@ extern void pagebuf_iomove( /* move data in/out of pagebuf */
caddr_t, /* data pointer */ caddr_t, /* data pointer */
page_buf_rw_t); /* direction */ page_buf_rw_t); /* direction */
static inline int pagebuf_iostrategy(page_buf_t *pb)
{
return pb->pb_strat ? pb->pb_strat(pb) : pagebuf_iorequest(pb);
}
static inline int pagebuf_geterror(page_buf_t *pb)
{
return pb ? pb->pb_error : ENOMEM;
}
/* Buffer Utility Routines */
extern caddr_t pagebuf_offset( /* pointer at offset in buffer */
page_buf_t *, /* buffer to offset into */
size_t); /* offset */
/* Pinning Buffer Storage in Memory */ /* Pinning Buffer Storage in Memory */
extern void pagebuf_pin( /* pin buffer in memory */ extern void pagebuf_pin( /* pin buffer in memory */
...@@ -369,33 +365,24 @@ extern void pagebuf_pin( /* pin buffer in memory */ ...@@ -369,33 +365,24 @@ extern void pagebuf_pin( /* pin buffer in memory */
extern void pagebuf_unpin( /* unpin buffered data */ extern void pagebuf_unpin( /* unpin buffered data */
page_buf_t *); /* buffer to unpin */ page_buf_t *); /* buffer to unpin */
extern int pagebuf_ispin( page_buf_t *); /* check if pagebuf is pinned */ extern int pagebuf_ispin( /* check if buffer is pinned */
page_buf_t *); /* buffer to check */
/* Reading and writing pages */
extern void pagebuf_delwri_dequeue(page_buf_t *); /* Delayed Write Buffer Routines */
#define PBDF_WAIT 0x01 #define PBDF_WAIT 0x01
#define PBDF_TRYLOCK 0x02 #define PBDF_TRYLOCK 0x02
extern void pagebuf_delwri_flush( extern void pagebuf_delwri_flush(
struct pb_target *, pb_target_t *,
unsigned long, unsigned long,
int *); int *);
extern int pagebuf_init(void); extern void pagebuf_delwri_dequeue(
extern void pagebuf_terminate(void); page_buf_t *);
static __inline__ int __pagebuf_iorequest(page_buf_t *pb) /* Buffer Daemon Setup Routines */
{
if (pb->pb_strat)
return pb->pb_strat(pb);
return pagebuf_iorequest(pb);
}
static __inline__ void pagebuf_run_queues(page_buf_t *pb) extern int pagebuf_init(void);
{ extern void pagebuf_terminate(void);
if (!pb || atomic_read(&pb->pb_io_remaining))
blk_run_queues();
}
#endif /* __PAGE_BUF_H__ */ #endif /* __PAGE_BUF_H__ */
...@@ -113,7 +113,8 @@ pagebuf_lock( ...@@ -113,7 +113,8 @@ pagebuf_lock(
ASSERT(pb->pb_flags & _PBF_LOCKABLE); ASSERT(pb->pb_flags & _PBF_LOCKABLE);
PB_TRACE(pb, PB_TRACE_REC(lock), 0); PB_TRACE(pb, PB_TRACE_REC(lock), 0);
pagebuf_run_queues(pb); if (atomic_read(&pb->pb_io_remaining))
blk_run_queues();
down(&pb->pb_sema); down(&pb->pb_sema);
PB_SET_OWNER(pb); PB_SET_OWNER(pb);
PB_TRACE(pb, PB_TRACE_REC(locked), 0); PB_TRACE(pb, PB_TRACE_REC(locked), 0);
......
...@@ -1718,6 +1718,7 @@ xfs_attr_node_get(xfs_da_args_t *args) ...@@ -1718,6 +1718,7 @@ xfs_attr_node_get(xfs_da_args_t *args)
int i; int i;
state = xfs_da_state_alloc(); state = xfs_da_state_alloc();
state->holeok = 1;
state->args = args; state->args = args;
state->mp = args->dp->i_mount; state->mp = args->dp->i_mount;
state->blocksize = state->mp->m_sb.sb_blocksize; state->blocksize = state->mp->m_sb.sb_blocksize;
......
...@@ -3810,7 +3810,7 @@ xfs_bmap_add_attrfork( ...@@ -3810,7 +3810,7 @@ xfs_bmap_add_attrfork(
xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE); xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
switch (ip->i_d.di_format) { switch (ip->i_d.di_format) {
case XFS_DINODE_FMT_DEV: case XFS_DINODE_FMT_DEV:
ip->i_d.di_forkoff = roundup(sizeof(dev_t), 8) >> 3; ip->i_d.di_forkoff = roundup(sizeof(xfs_dev_t), 8) >> 3;
break; break;
case XFS_DINODE_FMT_UUID: case XFS_DINODE_FMT_UUID:
ip->i_d.di_forkoff = roundup(sizeof(uuid_t), 8) >> 3; ip->i_d.di_forkoff = roundup(sizeof(uuid_t), 8) >> 3;
......
...@@ -215,21 +215,16 @@ extern inline xfs_caddr_t xfs_buf_offset(page_buf_t *bp, size_t offset) ...@@ -215,21 +215,16 @@ extern inline xfs_caddr_t xfs_buf_offset(page_buf_t *bp, size_t offset)
static inline int xfs_bawrite(void *mp, page_buf_t *bp) static inline int xfs_bawrite(void *mp, page_buf_t *bp)
{ {
int ret;
bp->pb_fspriv3 = mp; bp->pb_fspriv3 = mp;
bp->pb_strat = xfs_bdstrat_cb; bp->pb_strat = xfs_bdstrat_cb;
xfs_buf_undelay(bp); xfs_buf_undelay(bp);
if ((ret = pagebuf_iostart(bp, PBF_WRITE | PBF_ASYNC)) == 0) return pagebuf_iostart(bp, PBF_WRITE | PBF_ASYNC | PBF_RUN_QUEUES);
pagebuf_run_queues(bp);
return ret;
} }
static inline void xfs_buf_relse(page_buf_t *bp) static inline void xfs_buf_relse(page_buf_t *bp)
{ {
if ((bp->pb_flags & _PBF_LOCKABLE) && !bp->pb_relse) if ((bp->pb_flags & _PBF_LOCKABLE) && !bp->pb_relse)
pagebuf_unlock(bp); pagebuf_unlock(bp);
pagebuf_rele(bp); pagebuf_rele(bp);
} }
...@@ -263,23 +258,19 @@ static inline void xfs_buf_relse(page_buf_t *bp) ...@@ -263,23 +258,19 @@ static inline void xfs_buf_relse(page_buf_t *bp)
static inline int XFS_bwrite(page_buf_t *pb) static inline int XFS_bwrite(page_buf_t *pb)
{ {
int sync = (pb->pb_flags & PBF_ASYNC) == 0; int iowait = (pb->pb_flags & PBF_ASYNC) == 0;
int error; int error = 0;
pb->pb_flags |= PBF_SYNC; pb->pb_flags |= PBF_SYNC;
if (!iowait)
pb->pb_flags |= PBF_RUN_QUEUES;
xfs_buf_undelay(pb); xfs_buf_undelay(pb);
pagebuf_iostrategy(pb);
__pagebuf_iorequest(pb); if (iowait) {
if (sync) {
error = pagebuf_iowait(pb); error = pagebuf_iowait(pb);
xfs_buf_relse(pb); xfs_buf_relse(pb);
} else {
pagebuf_run_queues(pb);
error = 0;
} }
return error; return error;
} }
...@@ -320,4 +311,4 @@ static inline int xfs_bdwrite(void *mp, page_buf_t *bp) ...@@ -320,4 +311,4 @@ static inline int xfs_bdwrite(void *mp, page_buf_t *bp)
#define xfs_buf_get_noaddr(len, target) pagebuf_get_no_daddr((len), (target)) #define xfs_buf_get_noaddr(len, target) pagebuf_get_no_daddr((len), (target))
#define xfs_buf_free(bp) pagebuf_free(bp) #define xfs_buf_free(bp) pagebuf_free(bp)
#endif #endif /* __XFS_BUF_H__ */
/* /*
* Copyright (c) 2000-2002 Silicon Graphics, Inc. All Rights Reserved. * Copyright (c) 2000-2003 Silicon Graphics, Inc. All Rights Reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of version 2 of the GNU General Public License as * under the terms of version 2 of the GNU General Public License as
...@@ -1141,10 +1141,13 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result) ...@@ -1141,10 +1141,13 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result)
xfs_da_node_entry_t *btree; xfs_da_node_entry_t *btree;
xfs_dablk_t blkno; xfs_dablk_t blkno;
int probe, span, max, error, retval; int probe, span, max, error, retval;
xfs_daddr_t mappedbno;
xfs_dahash_t hashval; xfs_dahash_t hashval;
xfs_da_args_t *args; xfs_da_args_t *args;
args = state->args; args = state->args;
mappedbno = state->holeok ? -2 : -1;
/* /*
* Descend thru the B-tree searching each level for the right * Descend thru the B-tree searching each level for the right
* node to use, until the right hashval is found. * node to use, until the right hashval is found.
...@@ -1160,15 +1163,15 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result) ...@@ -1160,15 +1163,15 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result)
* Read the next node down in the tree. * Read the next node down in the tree.
*/ */
blk->blkno = blkno; blk->blkno = blkno;
error = xfs_da_read_buf(state->args->trans, state->args->dp, error = xfs_da_read_buf(args->trans, args->dp, blkno,
blkno, -1, &blk->bp, mappedbno, &blk->bp, args->whichfork);
state->args->whichfork); if (!error && unlikely(state->holeok && !blk->bp))
error = XFS_ERROR(ENOATTR); /* always attr here */
if (error) { if (error) {
blk->blkno = 0; blk->blkno = 0;
state->path.active--; state->path.active--;
return(error); return(error);
} }
ASSERT(blk->bp != NULL);
curr = blk->bp->data; curr = blk->bp->data;
ASSERT(INT_GET(curr->magic, ARCH_CONVERT) == XFS_DA_NODE_MAGIC || ASSERT(INT_GET(curr->magic, ARCH_CONVERT) == XFS_DA_NODE_MAGIC ||
INT_GET(curr->magic, ARCH_CONVERT) == XFS_DIRX_LEAF_MAGIC(state->mp) || INT_GET(curr->magic, ARCH_CONVERT) == XFS_DIRX_LEAF_MAGIC(state->mp) ||
...@@ -1187,7 +1190,7 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result) ...@@ -1187,7 +1190,7 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result)
*/ */
max = INT_GET(node->hdr.count, ARCH_CONVERT); max = INT_GET(node->hdr.count, ARCH_CONVERT);
probe = span = max / 2; probe = span = max / 2;
hashval = state->args->hashval; hashval = args->hashval;
for (btree = &node->btree[probe]; span > 4; for (btree = &node->btree[probe]; span > 4;
btree = &node->btree[probe]) { btree = &node->btree[probe]) {
span /= 2; span /= 2;
...@@ -1250,22 +1253,22 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result) ...@@ -1250,22 +1253,22 @@ xfs_da_node_lookup_int(xfs_da_state_t *state, int *result)
for (;;) { for (;;) {
if (blk->magic == XFS_DIR_LEAF_MAGIC) { if (blk->magic == XFS_DIR_LEAF_MAGIC) {
ASSERT(XFS_DIR_IS_V1(state->mp)); ASSERT(XFS_DIR_IS_V1(state->mp));
retval = xfs_dir_leaf_lookup_int(blk->bp, state->args, retval = xfs_dir_leaf_lookup_int(blk->bp, args,
&blk->index); &blk->index);
} else if (blk->magic == XFS_DIR2_LEAFN_MAGIC) { } else if (blk->magic == XFS_DIR2_LEAFN_MAGIC) {
ASSERT(XFS_DIR_IS_V2(state->mp)); ASSERT(XFS_DIR_IS_V2(state->mp));
retval = xfs_dir2_leafn_lookup_int(blk->bp, state->args, retval = xfs_dir2_leafn_lookup_int(blk->bp, args,
&blk->index, state); &blk->index, state);
} }
#ifdef __KERNEL__ #ifdef __KERNEL__
else if (blk->magic == XFS_ATTR_LEAF_MAGIC) { else if (blk->magic == XFS_ATTR_LEAF_MAGIC) {
retval = xfs_attr_leaf_lookup_int(blk->bp, state->args); retval = xfs_attr_leaf_lookup_int(blk->bp, args);
blk->index = state->args->index; blk->index = args->index;
state->args->blkno = blk->blkno; args->blkno = blk->blkno;
} }
#endif #endif
if (((retval == ENOENT) || (retval == ENOATTR)) && if (((retval == ENOENT) || (retval == ENOATTR)) &&
(blk->hashval == state->args->hashval)) { (blk->hashval == args->hashval)) {
error = xfs_da_path_shift(state, &state->path, 1, 1, error = xfs_da_path_shift(state, &state->path, 1, 1,
&retval); &retval);
if (error) if (error)
......
...@@ -185,14 +185,14 @@ typedef struct xfs_da_args { ...@@ -185,14 +185,14 @@ typedef struct xfs_da_args {
int index; /* index of attr of interest in blk */ int index; /* index of attr of interest in blk */
xfs_dablk_t rmtblkno; /* remote attr value starting blkno */ xfs_dablk_t rmtblkno; /* remote attr value starting blkno */
int rmtblkcnt; /* remote attr value block count */ int rmtblkcnt; /* remote attr value block count */
int rename; /* T/F: this is an atomic rename op */
xfs_dablk_t blkno2; /* blkno of 2nd attr leaf of interest */ xfs_dablk_t blkno2; /* blkno of 2nd attr leaf of interest */
int index2; /* index of 2nd attr in blk */ int index2; /* index of 2nd attr in blk */
xfs_dablk_t rmtblkno2; /* remote attr value starting blkno */ xfs_dablk_t rmtblkno2; /* remote attr value starting blkno */
int rmtblkcnt2; /* remote attr value block count */ int rmtblkcnt2; /* remote attr value block count */
int justcheck; /* check for ok with no space */ int justcheck : 1; /* T/F: check for ok with no space */
int addname; /* T/F: this is an add operation */ int rename : 1; /* T/F: this is an atomic rename op */
int oknoent; /* T/F: ok to return ENOENT, else die */ int addname : 1; /* T/F: this is an add operation */
int oknoent : 1; /* T/F: ok to return ENOENT, else die */
} xfs_da_args_t; } xfs_da_args_t;
/* /*
...@@ -253,6 +253,7 @@ typedef struct xfs_da_state { ...@@ -253,6 +253,7 @@ typedef struct xfs_da_state {
xfs_da_state_path_t path; /* search/split paths */ xfs_da_state_path_t path; /* search/split paths */
xfs_da_state_path_t altpath; /* alternate path for join */ xfs_da_state_path_t altpath; /* alternate path for join */
unsigned int inleaf : 1; /* insert into 1->lf, 0->splf */ unsigned int inleaf : 1; /* insert into 1->lf, 0->splf */
unsigned int holeok : 1; /* T/F: can deal with a hole */
unsigned int extravalid : 1; /* T/F: extrablk is in use */ unsigned int extravalid : 1; /* T/F: extrablk is in use */
unsigned int extraafter : 1; /* T/F: extrablk is after new */ unsigned int extraafter : 1; /* T/F: extrablk is after new */
xfs_da_state_blk_t extrablk; /* for double-splits on leafs */ xfs_da_state_blk_t extrablk; /* for double-splits on leafs */
......
...@@ -72,7 +72,8 @@ typedef struct xfs_dinode_core ...@@ -72,7 +72,8 @@ typedef struct xfs_dinode_core
__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; /* owner's project id */ __uint16_t di_projid; /* owner's project id */
__uint8_t di_pad[10]; /* unused, zeroed space */ __uint8_t di_pad[8]; /* unused, zeroed space */
__uint16_t di_flushiter; /* incremented on flush */
xfs_timestamp_t di_atime; /* time last accessed */ xfs_timestamp_t di_atime; /* time last accessed */
xfs_timestamp_t di_mtime; /* time last modified */ xfs_timestamp_t di_mtime; /* time last modified */
xfs_timestamp_t di_ctime; /* time created/inode modified */ xfs_timestamp_t di_ctime; /* time created/inode modified */
...@@ -89,6 +90,8 @@ typedef struct xfs_dinode_core ...@@ -89,6 +90,8 @@ typedef struct xfs_dinode_core
__uint32_t di_gen; /* generation number */ __uint32_t di_gen; /* generation number */
} xfs_dinode_core_t; } xfs_dinode_core_t;
#define DI_MAX_FLUSH 0xffff
typedef struct xfs_dinode typedef struct xfs_dinode
{ {
xfs_dinode_core_t di_core; xfs_dinode_core_t di_core;
......
...@@ -64,7 +64,7 @@ struct xfs_trans; ...@@ -64,7 +64,7 @@ struct xfs_trans;
* 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)0x7fffffff) #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)
/* /*
......
...@@ -560,14 +560,7 @@ xfs_dir_shortform_getdents(xfs_inode_t *dp, uio_t *uio, int *eofp, ...@@ -560,14 +560,7 @@ xfs_dir_shortform_getdents(xfs_inode_t *dp, uio_t *uio, int *eofp,
*/ */
if (sbp->seqno == 0 || sbp == sbuf) if (sbp->seqno == 0 || sbp == sbuf)
lastresid = uio->uio_resid; lastresid = uio->uio_resid;
/* XFS_PUT_COOKIE(p.cook, mp, 0, sbp[1].seqno, sbp[1].hash);
* NOTE! Linux "filldir" semantics require that the
* offset "cookie" be for this entry, not the
* next; all the actual shuffling to make it
* "look right" to the user is done in filldir.
*/
XFS_PUT_COOKIE(p.cook, mp, 0, sbp->seqno, sbp->hash);
#if XFS_BIG_FILESYSTEMS #if XFS_BIG_FILESYSTEMS
p.ino = sbp->ino + mp->m_inoadd; p.ino = sbp->ino + mp->m_inoadd;
#else #else
...@@ -575,9 +568,7 @@ xfs_dir_shortform_getdents(xfs_inode_t *dp, uio_t *uio, int *eofp, ...@@ -575,9 +568,7 @@ xfs_dir_shortform_getdents(xfs_inode_t *dp, uio_t *uio, int *eofp,
#endif #endif
p.name = sbp->name; p.name = sbp->name;
p.namelen = sbp->namelen; p.namelen = sbp->namelen;
retval = p.put(&p); retval = p.put(&p);
if (!p.done) { if (!p.done) {
uio->uio_offset = uio->uio_offset =
XFS_DA_MAKE_COOKIE(mp, 0, 0, sbp->hash); XFS_DA_MAKE_COOKIE(mp, 0, 0, sbp->hash);
...@@ -586,20 +577,12 @@ xfs_dir_shortform_getdents(xfs_inode_t *dp, uio_t *uio, int *eofp, ...@@ -586,20 +577,12 @@ xfs_dir_shortform_getdents(xfs_inode_t *dp, uio_t *uio, int *eofp,
xfs_dir_trace_g_du("sf: E-O-B", dp, uio); xfs_dir_trace_g_du("sf: E-O-B", dp, uio);
return retval; return retval;
} }
sbp++; sbp++;
} }
kmem_free(sbuf, sbsize); kmem_free(sbuf, sbsize);
XFS_PUT_COOKIE(p.cook, mp, 0, 0, XFS_DA_MAXHASH);
uio->uio_offset = p.cook.o; uio->uio_offset = p.cook.o;
*eofp = 1; *eofp = 1;
xfs_dir_trace_g_du("sf: E-O-F", dp, uio); xfs_dir_trace_g_du("sf: E-O-F", dp, uio);
return 0; return 0;
} }
...@@ -2070,16 +2053,6 @@ xfs_dir_leaf_getdents_int( ...@@ -2070,16 +2053,6 @@ xfs_dir_leaf_getdents_int(
return XFS_ERROR(EFSCORRUPTED); return XFS_ERROR(EFSCORRUPTED);
} }
thishash = INT_GET(entry->hashval, ARCH_CONVERT);
/*
* NOTE! Linux "filldir" semantics require that the
* offset "cookie" be for this entry, not the
* next; all the actual shuffling to make it
* "look right" to the user is done in filldir.
*/
XFS_PUT_COOKIE(p.cook, mp, bno, entno, thishash);
xfs_dir_trace_g_duc("leaf: middle cookie ", xfs_dir_trace_g_duc("leaf: middle cookie ",
dp, uio, p.cook.o); dp, uio, p.cook.o);
...@@ -2090,17 +2063,19 @@ xfs_dir_leaf_getdents_int( ...@@ -2090,17 +2063,19 @@ xfs_dir_leaf_getdents_int(
nextentno = entno + 1; nextentno = entno + 1;
else else
nextentno = 0; nextentno = 0;
XFS_PUT_COOKIE(p.cook, mp, bno, nextentno, nexthash);
xfs_dir_trace_g_duc("leaf: middle cookie ",
dp, uio, p.cook.o);
} else if (INT_GET(leaf->hdr.info.forw, ARCH_CONVERT)) { } else if ((thishash = INT_GET(leaf->hdr.info.forw,
ARCH_CONVERT))) {
xfs_dabuf_t *bp2; xfs_dabuf_t *bp2;
xfs_dir_leafblock_t *leaf2; xfs_dir_leafblock_t *leaf2;
ASSERT(nextda != -1); ASSERT(nextda != -1);
retval = xfs_da_read_buf(dp->i_transp, dp, retval = xfs_da_read_buf(dp->i_transp, dp, thishash,
INT_GET(leaf->hdr.info.forw, nextda, &bp2, XFS_DATA_FORK);
ARCH_CONVERT), nextda,
&bp2, XFS_DATA_FORK);
if (retval) if (retval)
return(retval); return(retval);
...@@ -2124,13 +2099,13 @@ xfs_dir_leaf_getdents_int( ...@@ -2124,13 +2099,13 @@ xfs_dir_leaf_getdents_int(
nexthash = INT_GET(leaf2->entries[0].hashval, nexthash = INT_GET(leaf2->entries[0].hashval,
ARCH_CONVERT); ARCH_CONVERT);
nextentno = -1; nextentno = -1;
XFS_PUT_COOKIE(p.cook, mp, thishash, 0, nexthash);
xfs_da_brelse(dp->i_transp, bp2); xfs_da_brelse(dp->i_transp, bp2);
xfs_dir_trace_g_duc("leaf: next blk cookie", xfs_dir_trace_g_duc("leaf: next blk cookie",
dp, uio, p.cook.o); dp, uio, p.cook.o);
} else { } else {
nextentno = -1; nextentno = -1;
nexthash = XFS_DA_MAXHASH; XFS_PUT_COOKIE(p.cook, mp, 0, 0, XFS_DA_MAXHASH);
} }
/* /*
...@@ -2147,7 +2122,8 @@ xfs_dir_leaf_getdents_int( ...@@ -2147,7 +2122,8 @@ xfs_dir_leaf_getdents_int(
* provided is big enough to handle it (see pv763517). * provided is big enough to handle it (see pv763517).
*/ */
#if (BITS_PER_LONG == 32) #if (BITS_PER_LONG == 32)
if (INT_GET(entry->hashval, ARCH_CONVERT) != lasthash) { if ((thishash = INT_GET(entry->hashval, ARCH_CONVERT))
!= lasthash) {
XFS_PUT_COOKIE(lastoffset, mp, bno, entno, thishash); XFS_PUT_COOKIE(lastoffset, mp, bno, entno, thishash);
lastresid = uio->uio_resid; lastresid = uio->uio_resid;
lasthash = thishash; lasthash = thishash;
...@@ -2156,6 +2132,7 @@ xfs_dir_leaf_getdents_int( ...@@ -2156,6 +2132,7 @@ xfs_dir_leaf_getdents_int(
dp, uio, p.cook.o); dp, uio, p.cook.o);
} }
#else #else
thishash = INT_GET(entry->hashval, ARCH_CONVERT);
XFS_PUT_COOKIE(lastoffset, mp, bno, entno, thishash); XFS_PUT_COOKIE(lastoffset, mp, bno, entno, thishash);
lastresid = uio->uio_resid; lastresid = uio->uio_resid;
#endif /* BITS_PER_LONG == 32 */ #endif /* BITS_PER_LONG == 32 */
...@@ -2187,8 +2164,6 @@ xfs_dir_leaf_getdents_int( ...@@ -2187,8 +2164,6 @@ xfs_dir_leaf_getdents_int(
} }
} }
XFS_PUT_COOKIE(p.cook, mp, 0, 0, nexthash);
uio->uio_offset = p.cook.o; uio->uio_offset = p.cook.o;
*eobp = 0; *eobp = 0;
......
...@@ -825,6 +825,8 @@ xfs_xlate_dinode_core( ...@@ -825,6 +825,8 @@ xfs_xlate_dinode_core(
sizeof(buf_core->di_pad)); sizeof(buf_core->di_pad));
} }
INT_XLATE(buf_core->di_flushiter, mem_core->di_flushiter, dir, arch);
INT_XLATE(buf_core->di_atime.t_sec, mem_core->di_atime.t_sec, INT_XLATE(buf_core->di_atime.t_sec, mem_core->di_atime.t_sec,
dir, arch); dir, arch);
INT_XLATE(buf_core->di_atime.t_nsec, mem_core->di_atime.t_nsec, INT_XLATE(buf_core->di_atime.t_nsec, mem_core->di_atime.t_nsec,
...@@ -956,6 +958,7 @@ xfs_iread( ...@@ -956,6 +958,7 @@ xfs_iread(
ip->i_d.di_magic = INT_GET(dip->di_core.di_magic, ARCH_CONVERT); ip->i_d.di_magic = INT_GET(dip->di_core.di_magic, ARCH_CONVERT);
ip->i_d.di_version = INT_GET(dip->di_core.di_version, ARCH_CONVERT); ip->i_d.di_version = INT_GET(dip->di_core.di_version, ARCH_CONVERT);
ip->i_d.di_gen = INT_GET(dip->di_core.di_gen, ARCH_CONVERT); ip->i_d.di_gen = INT_GET(dip->di_core.di_gen, ARCH_CONVERT);
ip->i_d.di_flushiter = INT_GET(dip->di_core.di_flushiter, ARCH_CONVERT);
/* /*
* Make sure to pull in the mode here as well in * Make sure to pull in the mode here as well in
* case the inode is released without being used. * case the inode is released without being used.
...@@ -3232,6 +3235,13 @@ xfs_iflush_int( ...@@ -3232,6 +3235,13 @@ xfs_iflush_int(
ip->i_ino, ip->i_d.di_forkoff, ip); ip->i_ino, ip->i_d.di_forkoff, ip);
goto corrupt_out; goto corrupt_out;
} }
/*
* bump the flush iteration count, used to detect flushes which
* postdate a log record during recovery.
*/
ip->i_d.di_flushiter++;
/* /*
* Copy the dirty parts of the inode into the on-disk * Copy the dirty parts of the inode into the on-disk
* inode. We always copy out the core of the inode, * inode. We always copy out the core of the inode,
...@@ -3241,6 +3251,10 @@ xfs_iflush_int( ...@@ -3241,6 +3251,10 @@ xfs_iflush_int(
xfs_xlate_dinode_core((xfs_caddr_t)&(dip->di_core), &(ip->i_d), xfs_xlate_dinode_core((xfs_caddr_t)&(dip->di_core), &(ip->i_d),
-1, ARCH_CONVERT); -1, ARCH_CONVERT);
/* Wrap, we never let the log put out DI_MAX_FLUSH */
if (ip->i_d.di_flushiter == DI_MAX_FLUSH)
ip->i_d.di_flushiter = 0;
/* /*
* If this is really an old format inode and the superblock version * If this is really an old format inode and the superblock version
* has not been updated to support only new format inodes, then * has not been updated to support only new format inodes, then
......
...@@ -2326,6 +2326,26 @@ xlog_recover_do_inode_trans( ...@@ -2326,6 +2326,26 @@ xlog_recover_do_inode_trans(
XFS_ERRLEVEL_LOW, mp); XFS_ERRLEVEL_LOW, mp);
return XFS_ERROR(EFSCORRUPTED); return XFS_ERROR(EFSCORRUPTED);
} }
/* Skip replay when the on disk inode is newer than the log one */
if (dicp->di_flushiter <
INT_GET(dip->di_core.di_flushiter, ARCH_CONVERT)) {
/*
* Deal with the wrap case, DI_MAX_FLUSH is less
* than smaller numbers
*/
if ((INT_GET(dip->di_core.di_flushiter, ARCH_CONVERT)
== DI_MAX_FLUSH) &&
(dicp->di_flushiter < (DI_MAX_FLUSH>>1))) {
/* do nothing */
} else {
xfs_buf_relse(bp);
return 0;
}
}
/* Take the opportunity to reset the flush iteration count */
dicp->di_flushiter = 0;
if (unlikely((dicp->di_mode & IFMT) == IFREG)) { if (unlikely((dicp->di_mode & IFMT) == IFREG)) {
if ((dicp->di_format != XFS_DINODE_FMT_EXTENTS) && if ((dicp->di_format != XFS_DINODE_FMT_EXTENTS) &&
(dicp->di_format != XFS_DINODE_FMT_BTREE)) { (dicp->di_format != XFS_DINODE_FMT_BTREE)) {
......
...@@ -1506,7 +1506,7 @@ xfs_syncsub( ...@@ -1506,7 +1506,7 @@ xfs_syncsub(
xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE); xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
error = xfs_trans_commit(tp, 0, NULL); error = xfs_trans_commit(tp, 0, NULL);
xfs_iunlock(ip, XFS_ILOCK_EXCL); xfs_iunlock(ip, XFS_ILOCK_EXCL);
xfs_log_force(mp, (xfs_lsn_t)0, XFS_LOG_FORCE); xfs_log_force(mp, (xfs_lsn_t)0, log_flags);
} }
/* /*
......
...@@ -2813,7 +2813,6 @@ xfs_mkdir( ...@@ -2813,7 +2813,6 @@ xfs_mkdir(
xfs_inode_t *cdp; /* inode of created dir */ xfs_inode_t *cdp; /* inode of created dir */
vnode_t *cvp; /* vnode of created dir */ vnode_t *cvp; /* vnode of created dir */
xfs_trans_t *tp; xfs_trans_t *tp;
xfs_dev_t rdev;
xfs_mount_t *mp; xfs_mount_t *mp;
int cancel_flags; int cancel_flags;
int error; int error;
...@@ -2912,10 +2911,9 @@ xfs_mkdir( ...@@ -2912,10 +2911,9 @@ xfs_mkdir(
/* /*
* create the directory inode. * create the directory inode.
*/ */
rdev = (vap->va_mask & XFS_AT_RDEV) ? vap->va_rdev : 0;
error = xfs_dir_ialloc(&tp, dp, error = xfs_dir_ialloc(&tp, dp,
MAKEIMODE(vap->va_type,vap->va_mode), 2, MAKEIMODE(vap->va_type,vap->va_mode), 2,
rdev, credp, prid, resblks > 0, 0, credp, prid, resblks > 0,
&cdp, NULL); &cdp, NULL);
if (error) { if (error) {
if (error == ENOSPC) if (error == ENOSPC)
...@@ -3336,7 +3334,6 @@ xfs_symlink( ...@@ -3336,7 +3334,6 @@ xfs_symlink(
xfs_inode_t *ip; xfs_inode_t *ip;
int error; int error;
int pathlen; int pathlen;
xfs_dev_t rdev;
xfs_bmap_free_t free_list; xfs_bmap_free_t free_list;
xfs_fsblock_t first_block; xfs_fsblock_t first_block;
boolean_t dp_joined_to_trans; boolean_t dp_joined_to_trans;
...@@ -3479,10 +3476,8 @@ xfs_symlink( ...@@ -3479,10 +3476,8 @@ xfs_symlink(
/* /*
* Allocate an inode for the symlink. * Allocate an inode for the symlink.
*/ */
rdev = (vap->va_mask & XFS_AT_RDEV) ? vap->va_rdev : 0;
error = xfs_dir_ialloc(&tp, dp, IFLNK | (vap->va_mode&~IFMT), error = xfs_dir_ialloc(&tp, dp, IFLNK | (vap->va_mode&~IFMT),
1, rdev, credp, prid, resblks > 0, &ip, NULL); 1, 0, credp, prid, resblks > 0, &ip, NULL);
if (error) { if (error) {
if (error == ENOSPC) if (error == ENOSPC)
goto error_return; goto error_return;
...@@ -3730,9 +3725,7 @@ xfs_inode_flush( ...@@ -3730,9 +3725,7 @@ xfs_inode_flush(
int flags) int flags)
{ {
xfs_inode_t *ip; xfs_inode_t *ip;
xfs_dinode_t *dip;
xfs_mount_t *mp; xfs_mount_t *mp;
xfs_buf_t *bp;
int error = 0; int error = 0;
ip = XFS_BHVTOI(bdp); ip = XFS_BHVTOI(bdp);
...@@ -3778,44 +3771,30 @@ xfs_inode_flush( ...@@ -3778,44 +3771,30 @@ xfs_inode_flush(
* now, they get caught later by xfs_sync. * now, they get caught later by xfs_sync.
*/ */
if (flags & FLUSH_INODE) { if (flags & FLUSH_INODE) {
if (xfs_ilock_nowait(ip, XFS_ILOCK_SHARED)) {
if ((xfs_ipincount(ip) == 0) && xfs_iflock_nowait(ip)) {
int flush_flags; int flush_flags;
#if 0 if (xfs_ipincount(ip))
/* not turning this on until some
* performance analysis is done
*/
if (flags & FLUSH_SYNC)
flush_flags = XFS_IFLUSH_SYNC;
else
#endif
flush_flags = XFS_IFLUSH_DELWRI_ELSE_ASYNC;
xfs_ifunlock(ip);
xfs_iunlock(ip, XFS_ILOCK_SHARED);
error = xfs_itobp(mp, NULL, ip, &dip, &bp, 0);
if (error)
return error;
xfs_buf_relse(bp);
if (xfs_ilock_nowait(ip, XFS_ILOCK_SHARED) == 0)
return EAGAIN; return EAGAIN;
if (xfs_ipincount(ip) || if (flags & FLUSH_SYNC) {
!xfs_iflock_nowait(ip)) { xfs_ilock(ip, XFS_ILOCK_SHARED);
xfs_iflock(ip);
} else if (xfs_ilock_nowait(ip, XFS_ILOCK_SHARED)) {
if (xfs_ipincount(ip) || !xfs_iflock_nowait(ip)) {
xfs_iunlock(ip, XFS_ILOCK_SHARED); xfs_iunlock(ip, XFS_ILOCK_SHARED);
return EAGAIN; return EAGAIN;
} }
error = xfs_iflush(ip, flush_flags);
} else { } else {
error = EAGAIN; return EAGAIN;
} }
if (flags & FLUSH_SYNC)
flush_flags = XFS_IFLUSH_SYNC;
else
flush_flags = XFS_IFLUSH_ASYNC;
error = xfs_iflush(ip, flush_flags);
xfs_iunlock(ip, XFS_ILOCK_SHARED); xfs_iunlock(ip, XFS_ILOCK_SHARED);
} else {
error = EAGAIN;
}
} }
return error; return error;
......
...@@ -3033,11 +3033,12 @@ xfs_prdinode_core(xfs_dinode_core_t *dip, int convert) ...@@ -3033,11 +3033,12 @@ xfs_prdinode_core(xfs_dinode_core_t *dip, int convert)
INT_GET(dip->di_format, convert), INT_GET(dip->di_format, convert),
xfs_fmtformat( xfs_fmtformat(
(xfs_dinode_fmt_t)INT_GET(dip->di_format, convert))); (xfs_dinode_fmt_t)INT_GET(dip->di_format, convert)));
kdb_printf("nlink 0x%x uid 0x%x gid 0x%x projid 0x%x\n", kdb_printf("nlink %d uid %d gid %d projid %d flushiter %u\n",
INT_GET(dip->di_nlink, convert), INT_GET(dip->di_nlink, convert),
INT_GET(dip->di_uid, convert), INT_GET(dip->di_uid, convert),
INT_GET(dip->di_gid, convert), INT_GET(dip->di_gid, convert),
(uint)INT_GET(dip->di_projid, convert)); (uint)INT_GET(dip->di_projid, convert),
(uint)INT_GET(dip->di_flushiter, convert));
kdb_printf("atime 0x%x:%x mtime 0x%x:%x ctime 0x%x:%x\n", kdb_printf("atime 0x%x:%x mtime 0x%x:%x ctime 0x%x:%x\n",
INT_GET(dip->di_atime.t_sec, convert), INT_GET(dip->di_atime.t_sec, convert),
INT_GET(dip->di_atime.t_nsec, convert), INT_GET(dip->di_atime.t_nsec, convert),
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment