aboutsummaryrefslogtreecommitdiffstats
path: root/fs/xfs
diff options
context:
space:
mode:
Diffstat (limited to 'fs/xfs')
-rw-r--r--fs/xfs/linux-2.6/xfs_super.c121
-rw-r--r--fs/xfs/linux-2.6/xfs_vfs.c3
-rw-r--r--fs/xfs/linux-2.6/xfs_vfs.h10
-rw-r--r--fs/xfs/xfs_mount.h6
4 files changed, 65 insertions, 75 deletions
diff --git a/fs/xfs/linux-2.6/xfs_super.c b/fs/xfs/linux-2.6/xfs_super.c
index e275b7a82bc1..3ce9426f57d8 100644
--- a/fs/xfs/linux-2.6/xfs_super.c
+++ b/fs/xfs/linux-2.6/xfs_super.c
@@ -457,9 +457,9 @@ xfs_fs_clear_inode(
457 */ 457 */
458STATIC void 458STATIC void
459xfs_syncd_queue_work( 459xfs_syncd_queue_work(
460 struct bhv_vfs *vfs, 460 struct xfs_mount *mp,
461 void *data, 461 void *data,
462 void (*syncer)(bhv_vfs_t *, void *)) 462 void (*syncer)(struct xfs_mount *, void *))
463{ 463{
464 struct bhv_vfs_sync_work *work; 464 struct bhv_vfs_sync_work *work;
465 465
@@ -467,11 +467,11 @@ xfs_syncd_queue_work(
467 INIT_LIST_HEAD(&work->w_list); 467 INIT_LIST_HEAD(&work->w_list);
468 work->w_syncer = syncer; 468 work->w_syncer = syncer;
469 work->w_data = data; 469 work->w_data = data;
470 work->w_vfs = vfs; 470 work->w_mount = mp;
471 spin_lock(&vfs->vfs_sync_lock); 471 spin_lock(&mp->m_sync_lock);
472 list_add_tail(&work->w_list, &vfs->vfs_sync_list); 472 list_add_tail(&work->w_list, &mp->m_sync_list);
473 spin_unlock(&vfs->vfs_sync_lock); 473 spin_unlock(&mp->m_sync_lock);
474 wake_up_process(vfs->vfs_sync_task); 474 wake_up_process(mp->m_sync_task);
475} 475}
476 476
477/* 477/*
@@ -482,22 +482,22 @@ xfs_syncd_queue_work(
482 */ 482 */
483STATIC void 483STATIC void
484xfs_flush_inode_work( 484xfs_flush_inode_work(
485 bhv_vfs_t *vfs, 485 struct xfs_mount *mp,
486 void *inode) 486 void *arg)
487{ 487{
488 filemap_flush(((struct inode *)inode)->i_mapping); 488 struct inode *inode = arg;
489 iput((struct inode *)inode); 489 filemap_flush(inode->i_mapping);
490 iput(inode);
490} 491}
491 492
492void 493void
493xfs_flush_inode( 494xfs_flush_inode(
494 xfs_inode_t *ip) 495 xfs_inode_t *ip)
495{ 496{
496 struct inode *inode = vn_to_inode(XFS_ITOV(ip)); 497 struct inode *inode = ip->i_vnode;
497 struct bhv_vfs *vfs = XFS_MTOVFS(ip->i_mount);
498 498
499 igrab(inode); 499 igrab(inode);
500 xfs_syncd_queue_work(vfs, inode, xfs_flush_inode_work); 500 xfs_syncd_queue_work(ip->i_mount, inode, xfs_flush_inode_work);
501 delay(msecs_to_jiffies(500)); 501 delay(msecs_to_jiffies(500));
502} 502}
503 503
@@ -507,11 +507,12 @@ xfs_flush_inode(
507 */ 507 */
508STATIC void 508STATIC void
509xfs_flush_device_work( 509xfs_flush_device_work(
510 bhv_vfs_t *vfs, 510 struct xfs_mount *mp,
511 void *inode) 511 void *arg)
512{ 512{
513 sync_blockdev(vfs->vfs_super->s_bdev); 513 struct inode *inode = arg;
514 iput((struct inode *)inode); 514 sync_blockdev(mp->m_vfsp->vfs_super->s_bdev);
515 iput(inode);
515} 516}
516 517
517void 518void
@@ -519,34 +520,33 @@ xfs_flush_device(
519 xfs_inode_t *ip) 520 xfs_inode_t *ip)
520{ 521{
521 struct inode *inode = vn_to_inode(XFS_ITOV(ip)); 522 struct inode *inode = vn_to_inode(XFS_ITOV(ip));
522 struct bhv_vfs *vfs = XFS_MTOVFS(ip->i_mount);
523 523
524 igrab(inode); 524 igrab(inode);
525 xfs_syncd_queue_work(vfs, inode, xfs_flush_device_work); 525 xfs_syncd_queue_work(ip->i_mount, inode, xfs_flush_device_work);
526 delay(msecs_to_jiffies(500)); 526 delay(msecs_to_jiffies(500));
527 xfs_log_force(ip->i_mount, (xfs_lsn_t)0, XFS_LOG_FORCE|XFS_LOG_SYNC); 527 xfs_log_force(ip->i_mount, (xfs_lsn_t)0, XFS_LOG_FORCE|XFS_LOG_SYNC);
528} 528}
529 529
530STATIC void 530STATIC void
531vfs_sync_worker( 531xfs_sync_worker(
532 bhv_vfs_t *vfsp, 532 struct xfs_mount *mp,
533 void *unused) 533 void *unused)
534{ 534{
535 int error; 535 int error;
536 536
537 if (!(XFS_VFSTOM(vfsp)->m_flags & XFS_MOUNT_RDONLY)) 537 if (!(mp->m_flags & XFS_MOUNT_RDONLY))
538 error = xfs_sync(XFS_VFSTOM(vfsp), SYNC_FSDATA | SYNC_BDFLUSH | \ 538 error = xfs_sync(mp, SYNC_FSDATA | SYNC_BDFLUSH | SYNC_ATTR |
539 SYNC_ATTR | SYNC_REFCACHE | SYNC_SUPER); 539 SYNC_REFCACHE | SYNC_SUPER);
540 vfsp->vfs_sync_seq++; 540 mp->m_sync_seq++;
541 wake_up(&vfsp->vfs_wait_single_sync_task); 541 wake_up(&mp->m_wait_single_sync_task);
542} 542}
543 543
544STATIC int 544STATIC int
545xfssyncd( 545xfssyncd(
546 void *arg) 546 void *arg)
547{ 547{
548 struct xfs_mount *mp = arg;
548 long timeleft; 549 long timeleft;
549 bhv_vfs_t *vfsp = (bhv_vfs_t *) arg;
550 bhv_vfs_sync_work_t *work, *n; 550 bhv_vfs_sync_work_t *work, *n;
551 LIST_HEAD (tmp); 551 LIST_HEAD (tmp);
552 552
@@ -556,31 +556,31 @@ xfssyncd(
556 timeleft = schedule_timeout_interruptible(timeleft); 556 timeleft = schedule_timeout_interruptible(timeleft);
557 /* swsusp */ 557 /* swsusp */
558 try_to_freeze(); 558 try_to_freeze();
559 if (kthread_should_stop() && list_empty(&vfsp->vfs_sync_list)) 559 if (kthread_should_stop() && list_empty(&mp->m_sync_list))
560 break; 560 break;
561 561
562 spin_lock(&vfsp->vfs_sync_lock); 562 spin_lock(&mp->m_sync_lock);
563 /* 563 /*
564 * We can get woken by laptop mode, to do a sync - 564 * We can get woken by laptop mode, to do a sync -
565 * that's the (only!) case where the list would be 565 * that's the (only!) case where the list would be
566 * empty with time remaining. 566 * empty with time remaining.
567 */ 567 */
568 if (!timeleft || list_empty(&vfsp->vfs_sync_list)) { 568 if (!timeleft || list_empty(&mp->m_sync_list)) {
569 if (!timeleft) 569 if (!timeleft)
570 timeleft = xfs_syncd_centisecs * 570 timeleft = xfs_syncd_centisecs *
571 msecs_to_jiffies(10); 571 msecs_to_jiffies(10);
572 INIT_LIST_HEAD(&vfsp->vfs_sync_work.w_list); 572 INIT_LIST_HEAD(&mp->m_sync_work.w_list);
573 list_add_tail(&vfsp->vfs_sync_work.w_list, 573 list_add_tail(&mp->m_sync_work.w_list,
574 &vfsp->vfs_sync_list); 574 &mp->m_sync_list);
575 } 575 }
576 list_for_each_entry_safe(work, n, &vfsp->vfs_sync_list, w_list) 576 list_for_each_entry_safe(work, n, &mp->m_sync_list, w_list)
577 list_move(&work->w_list, &tmp); 577 list_move(&work->w_list, &tmp);
578 spin_unlock(&vfsp->vfs_sync_lock); 578 spin_unlock(&mp->m_sync_lock);
579 579
580 list_for_each_entry_safe(work, n, &tmp, w_list) { 580 list_for_each_entry_safe(work, n, &tmp, w_list) {
581 (*work->w_syncer)(vfsp, work->w_data); 581 (*work->w_syncer)(mp, work->w_data);
582 list_del(&work->w_list); 582 list_del(&work->w_list);
583 if (work == &vfsp->vfs_sync_work) 583 if (work == &mp->m_sync_work)
584 continue; 584 continue;
585 kmem_free(work, sizeof(struct bhv_vfs_sync_work)); 585 kmem_free(work, sizeof(struct bhv_vfs_sync_work));
586 } 586 }
@@ -589,25 +589,6 @@ xfssyncd(
589 return 0; 589 return 0;
590} 590}
591 591
592STATIC int
593xfs_fs_start_syncd(
594 bhv_vfs_t *vfsp)
595{
596 vfsp->vfs_sync_work.w_syncer = vfs_sync_worker;
597 vfsp->vfs_sync_work.w_vfs = vfsp;
598 vfsp->vfs_sync_task = kthread_run(xfssyncd, vfsp, "xfssyncd");
599 if (IS_ERR(vfsp->vfs_sync_task))
600 return -PTR_ERR(vfsp->vfs_sync_task);
601 return 0;
602}
603
604STATIC void
605xfs_fs_stop_syncd(
606 bhv_vfs_t *vfsp)
607{
608 kthread_stop(vfsp->vfs_sync_task);
609}
610
611STATIC void 592STATIC void
612xfs_fs_put_super( 593xfs_fs_put_super(
613 struct super_block *sb) 594 struct super_block *sb)
@@ -616,7 +597,8 @@ xfs_fs_put_super(
616 struct xfs_mount *mp = XFS_M(sb); 597 struct xfs_mount *mp = XFS_M(sb);
617 int error; 598 int error;
618 599
619 xfs_fs_stop_syncd(vfsp); 600 kthread_stop(mp->m_sync_task);
601
620 xfs_sync(mp, SYNC_ATTR | SYNC_DELWRI); 602 xfs_sync(mp, SYNC_ATTR | SYNC_DELWRI);
621 error = xfs_unmount(mp, 0, NULL); 603 error = xfs_unmount(mp, 0, NULL);
622 if (error) { 604 if (error) {
@@ -641,7 +623,6 @@ xfs_fs_sync_super(
641 struct super_block *sb, 623 struct super_block *sb,
642 int wait) 624 int wait)
643{ 625{
644 bhv_vfs_t *vfsp = vfs_from_sb(sb);
645 struct xfs_mount *mp = XFS_M(sb); 626 struct xfs_mount *mp = XFS_M(sb);
646 int error; 627 int error;
647 int flags; 628 int flags;
@@ -663,22 +644,22 @@ xfs_fs_sync_super(
663 sb->s_dirt = 0; 644 sb->s_dirt = 0;
664 645
665 if (unlikely(laptop_mode)) { 646 if (unlikely(laptop_mode)) {
666 int prev_sync_seq = vfsp->vfs_sync_seq; 647 int prev_sync_seq = mp->m_sync_seq;
667 648
668 /* 649 /*
669 * The disk must be active because we're syncing. 650 * The disk must be active because we're syncing.
670 * We schedule xfssyncd now (now that the disk is 651 * We schedule xfssyncd now (now that the disk is
671 * active) instead of later (when it might not be). 652 * active) instead of later (when it might not be).
672 */ 653 */
673 wake_up_process(vfsp->vfs_sync_task); 654 wake_up_process(mp->m_sync_task);
674 /* 655 /*
675 * We have to wait for the sync iteration to complete. 656 * We have to wait for the sync iteration to complete.
676 * If we don't, the disk activity caused by the sync 657 * If we don't, the disk activity caused by the sync
677 * will come after the sync is completed, and that 658 * will come after the sync is completed, and that
678 * triggers another sync from laptop mode. 659 * triggers another sync from laptop mode.
679 */ 660 */
680 wait_event(vfsp->vfs_wait_single_sync_task, 661 wait_event(mp->m_wait_single_sync_task,
681 vfsp->vfs_sync_seq != prev_sync_seq); 662 mp->m_sync_seq != prev_sync_seq);
682 } 663 }
683 664
684 return -error; 665 return -error;
@@ -790,6 +771,11 @@ xfs_fs_fill_super(
790 int error; 771 int error;
791 772
792 mp = xfs_mount_init(); 773 mp = xfs_mount_init();
774
775 INIT_LIST_HEAD(&mp->m_sync_list);
776 spin_lock_init(&mp->m_sync_lock);
777 init_waitqueue_head(&mp->m_wait_single_sync_task);
778
793 mp->m_vfsp = vfsp; 779 mp->m_vfsp = vfsp;
794 vfsp->vfs_mount = mp; 780 vfsp->vfs_mount = mp;
795 781
@@ -834,8 +820,15 @@ xfs_fs_fill_super(
834 error = EINVAL; 820 error = EINVAL;
835 goto fail_vnrele; 821 goto fail_vnrele;
836 } 822 }
837 if ((error = xfs_fs_start_syncd(vfsp))) 823
824 mp->m_sync_work.w_syncer = xfs_sync_worker;
825 mp->m_sync_work.w_mount = mp;
826 mp->m_sync_task = kthread_run(xfssyncd, mp, "xfssyncd");
827 if (IS_ERR(mp->m_sync_task)) {
828 error = -PTR_ERR(mp->m_sync_task);
838 goto fail_vnrele; 829 goto fail_vnrele;
830 }
831
839 vn_trace_exit(XFS_I(sb->s_root->d_inode), __FUNCTION__, 832 vn_trace_exit(XFS_I(sb->s_root->d_inode), __FUNCTION__,
840 (inst_t *)__return_address); 833 (inst_t *)__return_address);
841 834
diff --git a/fs/xfs/linux-2.6/xfs_vfs.c b/fs/xfs/linux-2.6/xfs_vfs.c
index b098e0903353..97ce2eab69b0 100644
--- a/fs/xfs/linux-2.6/xfs_vfs.c
+++ b/fs/xfs/linux-2.6/xfs_vfs.c
@@ -204,9 +204,6 @@ vfs_allocate(
204 struct bhv_vfs *vfsp; 204 struct bhv_vfs *vfsp;
205 205
206 vfsp = kmem_zalloc(sizeof(bhv_vfs_t), KM_SLEEP); 206 vfsp = kmem_zalloc(sizeof(bhv_vfs_t), KM_SLEEP);
207 INIT_LIST_HEAD(&vfsp->vfs_sync_list);
208 spin_lock_init(&vfsp->vfs_sync_lock);
209 init_waitqueue_head(&vfsp->vfs_wait_single_sync_task);
210 207
211 vfsp->vfs_super = sb; 208 vfsp->vfs_super = sb;
212 sb->s_fs_info = vfsp; 209 sb->s_fs_info = vfsp;
diff --git a/fs/xfs/linux-2.6/xfs_vfs.h b/fs/xfs/linux-2.6/xfs_vfs.h
index 5c4996356226..51906192e70a 100644
--- a/fs/xfs/linux-2.6/xfs_vfs.h
+++ b/fs/xfs/linux-2.6/xfs_vfs.h
@@ -36,20 +36,14 @@ typedef struct kstatfs bhv_statvfs_t;
36 36
37typedef struct bhv_vfs_sync_work { 37typedef struct bhv_vfs_sync_work {
38 struct list_head w_list; 38 struct list_head w_list;
39 struct bhv_vfs *w_vfs; 39 struct xfs_mount *w_mount;
40 void *w_data; /* syncer routine argument */ 40 void *w_data; /* syncer routine argument */
41 void (*w_syncer)(struct bhv_vfs *, void *); 41 void (*w_syncer)(struct xfs_mount *, void *);
42} bhv_vfs_sync_work_t; 42} bhv_vfs_sync_work_t;
43 43
44typedef struct bhv_vfs { 44typedef struct bhv_vfs {
45 struct xfs_mount *vfs_mount; 45 struct xfs_mount *vfs_mount;
46 struct super_block *vfs_super; /* generic superblock pointer */ 46 struct super_block *vfs_super; /* generic superblock pointer */
47 struct task_struct *vfs_sync_task; /* generalised sync thread */
48 bhv_vfs_sync_work_t vfs_sync_work; /* work item for VFS_SYNC */
49 struct list_head vfs_sync_list; /* sync thread work item list */
50 spinlock_t vfs_sync_lock; /* work item list lock */
51 int vfs_sync_seq; /* sync thread generation no. */
52 wait_queue_head_t vfs_wait_single_sync_task;
53} bhv_vfs_t; 47} bhv_vfs_t;
54 48
55#define SYNC_ATTR 0x0001 /* sync attributes */ 49#define SYNC_ATTR 0x0001 /* sync attributes */
diff --git a/fs/xfs/xfs_mount.h b/fs/xfs/xfs_mount.h
index f37060bcde53..94f73c523e5f 100644
--- a/fs/xfs/xfs_mount.h
+++ b/fs/xfs/xfs_mount.h
@@ -433,6 +433,12 @@ typedef struct xfs_mount {
433 struct mutex m_icsb_mutex; /* balancer sync lock */ 433 struct mutex m_icsb_mutex; /* balancer sync lock */
434#endif 434#endif
435 struct xfs_mru_cache *m_filestream; /* per-mount filestream data */ 435 struct xfs_mru_cache *m_filestream; /* per-mount filestream data */
436 struct task_struct *m_sync_task; /* generalised sync thread */
437 bhv_vfs_sync_work_t m_sync_work; /* work item for VFS_SYNC */
438 struct list_head m_sync_list; /* sync thread work item list */
439 spinlock_t m_sync_lock; /* work item list lock */
440 int m_sync_seq; /* sync thread generation no. */
441 wait_queue_head_t m_wait_single_sync_task;
436} xfs_mount_t; 442} xfs_mount_t;
437 443
438/* 444/*