aboutsummaryrefslogtreecommitdiffstats
path: root/block/as-iosched.c
diff options
context:
space:
mode:
Diffstat (limited to 'block/as-iosched.c')
-rw-r--r--block/as-iosched.c34
1 files changed, 28 insertions, 6 deletions
diff --git a/block/as-iosched.c b/block/as-iosched.c
index cb5e53b05c7c..b201d16a7102 100644
--- a/block/as-iosched.c
+++ b/block/as-iosched.c
@@ -170,9 +170,11 @@ static void free_as_io_context(struct as_io_context *aic)
170 170
171static void as_trim(struct io_context *ioc) 171static void as_trim(struct io_context *ioc)
172{ 172{
173 spin_lock(&ioc->lock);
173 if (ioc->aic) 174 if (ioc->aic)
174 free_as_io_context(ioc->aic); 175 free_as_io_context(ioc->aic);
175 ioc->aic = NULL; 176 ioc->aic = NULL;
177 spin_unlock(&ioc->lock);
176} 178}
177 179
178/* Called when the task exits */ 180/* Called when the task exits */
@@ -462,7 +464,9 @@ static void as_antic_timeout(unsigned long data)
462 spin_lock_irqsave(q->queue_lock, flags); 464 spin_lock_irqsave(q->queue_lock, flags);
463 if (ad->antic_status == ANTIC_WAIT_REQ 465 if (ad->antic_status == ANTIC_WAIT_REQ
464 || ad->antic_status == ANTIC_WAIT_NEXT) { 466 || ad->antic_status == ANTIC_WAIT_NEXT) {
465 struct as_io_context *aic = ad->io_context->aic; 467 struct as_io_context *aic;
468 spin_lock(&ad->io_context->lock);
469 aic = ad->io_context->aic;
466 470
467 ad->antic_status = ANTIC_FINISHED; 471 ad->antic_status = ANTIC_FINISHED;
468 kblockd_schedule_work(&ad->antic_work); 472 kblockd_schedule_work(&ad->antic_work);
@@ -475,6 +479,7 @@ static void as_antic_timeout(unsigned long data)
475 /* process not "saved" by a cooperating request */ 479 /* process not "saved" by a cooperating request */
476 ad->exit_no_coop = (7*ad->exit_no_coop + 256)/8; 480 ad->exit_no_coop = (7*ad->exit_no_coop + 256)/8;
477 } 481 }
482 spin_unlock(&ad->io_context->lock);
478 } 483 }
479 spin_unlock_irqrestore(q->queue_lock, flags); 484 spin_unlock_irqrestore(q->queue_lock, flags);
480} 485}
@@ -635,9 +640,11 @@ static int as_can_break_anticipation(struct as_data *ad, struct request *rq)
635 640
636 ioc = ad->io_context; 641 ioc = ad->io_context;
637 BUG_ON(!ioc); 642 BUG_ON(!ioc);
643 spin_lock(&ioc->lock);
638 644
639 if (rq && ioc == RQ_IOC(rq)) { 645 if (rq && ioc == RQ_IOC(rq)) {
640 /* request from same process */ 646 /* request from same process */
647 spin_unlock(&ioc->lock);
641 return 1; 648 return 1;
642 } 649 }
643 650
@@ -646,20 +653,25 @@ static int as_can_break_anticipation(struct as_data *ad, struct request *rq)
646 * In this situation status should really be FINISHED, 653 * In this situation status should really be FINISHED,
647 * however the timer hasn't had the chance to run yet. 654 * however the timer hasn't had the chance to run yet.
648 */ 655 */
656 spin_unlock(&ioc->lock);
649 return 1; 657 return 1;
650 } 658 }
651 659
652 aic = ioc->aic; 660 aic = ioc->aic;
653 if (!aic) 661 if (!aic) {
662 spin_unlock(&ioc->lock);
654 return 0; 663 return 0;
664 }
655 665
656 if (atomic_read(&aic->nr_queued) > 0) { 666 if (atomic_read(&aic->nr_queued) > 0) {
657 /* process has more requests queued */ 667 /* process has more requests queued */
668 spin_unlock(&ioc->lock);
658 return 1; 669 return 1;
659 } 670 }
660 671
661 if (atomic_read(&aic->nr_dispatched) > 0) { 672 if (atomic_read(&aic->nr_dispatched) > 0) {
662 /* process has more requests dispatched */ 673 /* process has more requests dispatched */
674 spin_unlock(&ioc->lock);
663 return 1; 675 return 1;
664 } 676 }
665 677
@@ -680,6 +692,7 @@ static int as_can_break_anticipation(struct as_data *ad, struct request *rq)
680 } 692 }
681 693
682 as_update_iohist(ad, aic, rq); 694 as_update_iohist(ad, aic, rq);
695 spin_unlock(&ioc->lock);
683 return 1; 696 return 1;
684 } 697 }
685 698
@@ -688,20 +701,27 @@ static int as_can_break_anticipation(struct as_data *ad, struct request *rq)
688 if (aic->ttime_samples == 0) 701 if (aic->ttime_samples == 0)
689 ad->exit_prob = (7*ad->exit_prob + 256)/8; 702 ad->exit_prob = (7*ad->exit_prob + 256)/8;
690 703
691 if (ad->exit_no_coop > 128) 704 if (ad->exit_no_coop > 128) {
705 spin_unlock(&ioc->lock);
692 return 1; 706 return 1;
707 }
693 } 708 }
694 709
695 if (aic->ttime_samples == 0) { 710 if (aic->ttime_samples == 0) {
696 if (ad->new_ttime_mean > ad->antic_expire) 711 if (ad->new_ttime_mean > ad->antic_expire) {
712 spin_unlock(&ioc->lock);
697 return 1; 713 return 1;
698 if (ad->exit_prob * ad->exit_no_coop > 128*256) 714 }
715 if (ad->exit_prob * ad->exit_no_coop > 128*256) {
716 spin_unlock(&ioc->lock);
699 return 1; 717 return 1;
718 }
700 } else if (aic->ttime_mean > ad->antic_expire) { 719 } else if (aic->ttime_mean > ad->antic_expire) {
701 /* the process thinks too much between requests */ 720 /* the process thinks too much between requests */
721 spin_unlock(&ioc->lock);
702 return 1; 722 return 1;
703 } 723 }
704 724 spin_unlock(&ioc->lock);
705 return 0; 725 return 0;
706} 726}
707 727
@@ -1255,7 +1275,9 @@ static void as_merged_requests(struct request_queue *q, struct request *req,
1255 * Don't copy here but swap, because when anext is 1275 * Don't copy here but swap, because when anext is
1256 * removed below, it must contain the unused context 1276 * removed below, it must contain the unused context
1257 */ 1277 */
1278 double_spin_lock(&rioc->lock, &nioc->lock, rioc < nioc);
1258 swap_io_context(&rioc, &nioc); 1279 swap_io_context(&rioc, &nioc);
1280 double_spin_unlock(&rioc->lock, &nioc->lock, rioc < nioc);
1259 } 1281 }
1260 } 1282 }
1261 1283