aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorTejun Heo <tj@kernel.org>2009-05-07 22:54:01 -0400
committerJens Axboe <jens.axboe@oracle.com>2009-05-11 03:52:15 -0400
commit5b36ad6000ddea390aca3c3b67ead7616ace2ffc (patch)
treededca03fffd6729487fbd9bc3959428038e0cdfb /drivers
parent9a8d23d8855e554fc5887f14cb008b55c4300ccc (diff)
mg_disk: dequeue and track in-flight request
mg_disk has at most single request in flight per device. Till now, whenever it needs to access the in-flight request it called elv_next_request(). This patch makes mg_disk track the in-flight request directly using mg_host->req and dequeue it when processing starts. q->queuedata is set to mg_host so that mg_host can be determined without fetching request from the queue. [ Impact: dequeue in-flight request, one elv_next_request() per request ] Signed-off-by: Tejun Heo <tj@kernel.org> Cc: unsik Kim <donari75@gmail.com> Signed-off-by: Jens Axboe <jens.axboe@oracle.com>
Diffstat (limited to 'drivers')
-rw-r--r--drivers/block/mg_disk.c109
1 files changed, 59 insertions, 50 deletions
diff --git a/drivers/block/mg_disk.c b/drivers/block/mg_disk.c
index be323880f24a..1ca5d1423fa3 100644
--- a/drivers/block/mg_disk.c
+++ b/drivers/block/mg_disk.c
@@ -135,6 +135,7 @@ struct mg_host {
135 struct device *dev; 135 struct device *dev;
136 136
137 struct request_queue *breq; 137 struct request_queue *breq;
138 struct request *req;
138 spinlock_t lock; 139 spinlock_t lock;
139 struct gendisk *gd; 140 struct gendisk *gd;
140 141
@@ -171,17 +172,27 @@ struct mg_host {
171 172
172static void mg_request(struct request_queue *); 173static void mg_request(struct request_queue *);
173 174
175static bool mg_end_request(struct mg_host *host, int err, unsigned int nr_bytes)
176{
177 if (__blk_end_request(host->req, err, nr_bytes))
178 return true;
179
180 host->req = NULL;
181 return false;
182}
183
184static bool mg_end_request_cur(struct mg_host *host, int err)
185{
186 return mg_end_request(host, err, blk_rq_cur_bytes(host->req));
187}
188
174static void mg_dump_status(const char *msg, unsigned int stat, 189static void mg_dump_status(const char *msg, unsigned int stat,
175 struct mg_host *host) 190 struct mg_host *host)
176{ 191{
177 char *name = MG_DISK_NAME; 192 char *name = MG_DISK_NAME;
178 struct request *req;
179 193
180 if (host->breq) { 194 if (host->req)
181 req = elv_next_request(host->breq); 195 name = host->req->rq_disk->disk_name;
182 if (req)
183 name = req->rq_disk->disk_name;
184 }
185 196
186 printk(KERN_ERR "%s: %s: status=0x%02x { ", name, msg, stat & 0xff); 197 printk(KERN_ERR "%s: %s: status=0x%02x { ", name, msg, stat & 0xff);
187 if (stat & ATA_BUSY) 198 if (stat & ATA_BUSY)
@@ -217,13 +228,9 @@ static void mg_dump_status(const char *msg, unsigned int stat,
217 printk("AddrMarkNotFound "); 228 printk("AddrMarkNotFound ");
218 printk("}"); 229 printk("}");
219 if (host->error & (ATA_BBK | ATA_UNC | ATA_IDNF | ATA_AMNF)) { 230 if (host->error & (ATA_BBK | ATA_UNC | ATA_IDNF | ATA_AMNF)) {
220 if (host->breq) { 231 if (host->req)
221 req = elv_next_request(host->breq); 232 printk(", sector=%u",
222 if (req) 233 (unsigned int)blk_rq_pos(host->req));
223 printk(", sector=%u",
224 (unsigned int)blk_rq_pos(req));
225 }
226
227 } 234 }
228 printk("\n"); 235 printk("\n");
229 } 236 }
@@ -453,11 +460,10 @@ static int mg_disk_init(struct mg_host *host)
453 460
454static void mg_bad_rw_intr(struct mg_host *host) 461static void mg_bad_rw_intr(struct mg_host *host)
455{ 462{
456 struct request *req = elv_next_request(host->breq); 463 if (host->req)
457 if (req != NULL) 464 if (++host->req->errors >= MG_MAX_ERRORS ||
458 if (++req->errors >= MG_MAX_ERRORS || 465 host->error == MG_ERR_TIMEOUT)
459 host->error == MG_ERR_TIMEOUT) 466 mg_end_request_cur(host, -EIO);
460 __blk_end_request_cur(req, -EIO);
461} 467}
462 468
463static unsigned int mg_out(struct mg_host *host, 469static unsigned int mg_out(struct mg_host *host,
@@ -515,7 +521,7 @@ static void mg_read(struct request *req)
515 521
516 outb(MG_CMD_RD_CONF, (unsigned long)host->dev_base + 522 outb(MG_CMD_RD_CONF, (unsigned long)host->dev_base +
517 MG_REG_COMMAND); 523 MG_REG_COMMAND);
518 } while (__blk_end_request(req, 0, MG_SECTOR_SIZE)); 524 } while (mg_end_request(host, 0, MG_SECTOR_SIZE));
519} 525}
520 526
521static void mg_write(struct request *req) 527static void mg_write(struct request *req)
@@ -545,14 +551,14 @@ static void mg_write(struct request *req)
545 551
546 outb(MG_CMD_WR_CONF, (unsigned long)host->dev_base + 552 outb(MG_CMD_WR_CONF, (unsigned long)host->dev_base +
547 MG_REG_COMMAND); 553 MG_REG_COMMAND);
548 } while (__blk_end_request(req, 0, MG_SECTOR_SIZE)); 554 } while (mg_end_request(host, 0, MG_SECTOR_SIZE));
549} 555}
550 556
551static void mg_read_intr(struct mg_host *host) 557static void mg_read_intr(struct mg_host *host)
552{ 558{
559 struct request *req = host->req;
553 u32 i; 560 u32 i;
554 u16 *buff; 561 u16 *buff;
555 struct request *req;
556 562
557 /* check status */ 563 /* check status */
558 do { 564 do {
@@ -571,7 +577,6 @@ static void mg_read_intr(struct mg_host *host)
571 577
572ok_to_read: 578ok_to_read:
573 /* get current segment of request */ 579 /* get current segment of request */
574 req = elv_next_request(host->breq);
575 buff = (u16 *)req->buffer; 580 buff = (u16 *)req->buffer;
576 581
577 /* read 1 sector */ 582 /* read 1 sector */
@@ -585,7 +590,7 @@ ok_to_read:
585 /* send read confirm */ 590 /* send read confirm */
586 outb(MG_CMD_RD_CONF, (unsigned long)host->dev_base + MG_REG_COMMAND); 591 outb(MG_CMD_RD_CONF, (unsigned long)host->dev_base + MG_REG_COMMAND);
587 592
588 if (__blk_end_request(req, 0, MG_SECTOR_SIZE)) { 593 if (mg_end_request(host, 0, MG_SECTOR_SIZE)) {
589 /* set handler if read remains */ 594 /* set handler if read remains */
590 host->mg_do_intr = mg_read_intr; 595 host->mg_do_intr = mg_read_intr;
591 mod_timer(&host->timer, jiffies + 3 * HZ); 596 mod_timer(&host->timer, jiffies + 3 * HZ);
@@ -595,14 +600,11 @@ ok_to_read:
595 600
596static void mg_write_intr(struct mg_host *host) 601static void mg_write_intr(struct mg_host *host)
597{ 602{
603 struct request *req = host->req;
598 u32 i, j; 604 u32 i, j;
599 u16 *buff; 605 u16 *buff;
600 struct request *req;
601 bool rem; 606 bool rem;
602 607
603 /* get current segment of request */
604 req = elv_next_request(host->breq);
605
606 /* check status */ 608 /* check status */
607 do { 609 do {
608 i = inb((unsigned long)host->dev_base + MG_REG_STATUS); 610 i = inb((unsigned long)host->dev_base + MG_REG_STATUS);
@@ -619,7 +621,7 @@ static void mg_write_intr(struct mg_host *host)
619 return; 621 return;
620 622
621ok_to_write: 623ok_to_write:
622 if ((rem = __blk_end_request(req, 0, MG_SECTOR_SIZE))) { 624 if ((rem = mg_end_request(host, 0, MG_SECTOR_SIZE))) {
623 /* write 1 sector and set handler if remains */ 625 /* write 1 sector and set handler if remains */
624 buff = (u16 *)req->buffer; 626 buff = (u16 *)req->buffer;
625 for (j = 0; j < MG_STORAGE_BUFFER_SIZE >> 1; j++) { 627 for (j = 0; j < MG_STORAGE_BUFFER_SIZE >> 1; j++) {
@@ -644,44 +646,47 @@ void mg_times_out(unsigned long data)
644{ 646{
645 struct mg_host *host = (struct mg_host *)data; 647 struct mg_host *host = (struct mg_host *)data;
646 char *name; 648 char *name;
647 struct request *req;
648 649
649 spin_lock_irq(&host->lock); 650 spin_lock_irq(&host->lock);
650 651
651 req = elv_next_request(host->breq); 652 if (!host->req)
652 if (!req)
653 goto out_unlock; 653 goto out_unlock;
654 654
655 host->mg_do_intr = NULL; 655 host->mg_do_intr = NULL;
656 656
657 name = req->rq_disk->disk_name; 657 name = host->req->rq_disk->disk_name;
658 printk(KERN_DEBUG "%s: timeout\n", name); 658 printk(KERN_DEBUG "%s: timeout\n", name);
659 659
660 host->error = MG_ERR_TIMEOUT; 660 host->error = MG_ERR_TIMEOUT;
661 mg_bad_rw_intr(host); 661 mg_bad_rw_intr(host);
662 662
663 mg_request(host->breq);
664out_unlock: 663out_unlock:
664 mg_request(host->breq);
665 spin_unlock_irq(&host->lock); 665 spin_unlock_irq(&host->lock);
666} 666}
667 667
668static void mg_request_poll(struct request_queue *q) 668static void mg_request_poll(struct request_queue *q)
669{ 669{
670 struct request *req; 670 struct mg_host *host = q->queuedata;
671 struct mg_host *host;
672 671
673 while ((req = elv_next_request(q)) != NULL) { 672 while (1) {
674 host = req->rq_disk->private_data; 673 if (!host->req) {
674 host->req = elv_next_request(q);
675 if (host->req)
676 blkdev_dequeue_request(host->req);
677 else
678 break;
679 }
675 680
676 if (unlikely(!blk_fs_request(req))) { 681 if (unlikely(!blk_fs_request(host->req))) {
677 __blk_end_request_cur(req, -EIO); 682 mg_end_request_cur(host, -EIO);
678 continue; 683 continue;
679 } 684 }
680 685
681 if (rq_data_dir(req) == READ) 686 if (rq_data_dir(host->req) == READ)
682 mg_read(req); 687 mg_read(host->req);
683 else 688 else
684 mg_write(req); 689 mg_write(host->req);
685 } 690 }
686} 691}
687 692
@@ -733,16 +738,19 @@ static unsigned int mg_issue_req(struct request *req,
733/* This function also called from IRQ context */ 738/* This function also called from IRQ context */
734static void mg_request(struct request_queue *q) 739static void mg_request(struct request_queue *q)
735{ 740{
741 struct mg_host *host = q->queuedata;
736 struct request *req; 742 struct request *req;
737 struct mg_host *host;
738 u32 sect_num, sect_cnt; 743 u32 sect_num, sect_cnt;
739 744
740 while (1) { 745 while (1) {
741 req = elv_next_request(q); 746 if (!host->req) {
742 if (!req) 747 host->req = elv_next_request(q);
743 return; 748 if (host->req)
744 749 blkdev_dequeue_request(host->req);
745 host = req->rq_disk->private_data; 750 else
751 break;
752 }
753 req = host->req;
746 754
747 /* check unwanted request call */ 755 /* check unwanted request call */
748 if (host->mg_do_intr) 756 if (host->mg_do_intr)
@@ -762,12 +770,12 @@ static void mg_request(struct request_queue *q)
762 "%s: bad access: sector=%d, count=%d\n", 770 "%s: bad access: sector=%d, count=%d\n",
763 req->rq_disk->disk_name, 771 req->rq_disk->disk_name,
764 sect_num, sect_cnt); 772 sect_num, sect_cnt);
765 __blk_end_request_cur(req, -EIO); 773 mg_end_request_cur(host, -EIO);
766 continue; 774 continue;
767 } 775 }
768 776
769 if (unlikely(!blk_fs_request(req))) { 777 if (unlikely(!blk_fs_request(req))) {
770 __blk_end_request_cur(req, -EIO); 778 mg_end_request_cur(host, -EIO);
771 continue; 779 continue;
772 } 780 }
773 781
@@ -981,6 +989,7 @@ static int mg_probe(struct platform_device *plat_dev)
981 __func__, __LINE__); 989 __func__, __LINE__);
982 goto probe_err_5; 990 goto probe_err_5;
983 } 991 }
992 host->breq->queuedata = host;
984 993
985 /* mflash is random device, thanx for the noop */ 994 /* mflash is random device, thanx for the noop */
986 elevator_exit(host->breq->elevator); 995 elevator_exit(host->breq->elevator);