aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorOGAWA Hirofumi <hirofumi@mail.parknet.co.jp>2006-04-18 13:18:31 -0400
committerJens Axboe <axboe@suse.de>2006-04-18 13:18:31 -0400
commitbe3b075354e170368a0d29558cae492205e80a64 (patch)
treeb37af91addb8d214b9010774f5cf31538a501267
parentdbecf3ab40b5a6cc4499543778cd9f9682c0abad (diff)
[PATCH] cfq: Further rbtree traversal and cfq_exit_queue() race fix
In current code, we are re-reading cic->key after dead cic->key check. So, in theory, it may really re-read *after* cfq_exit_queue() seted NULL. To avoid race, we copy it to stack, then use it. With this change, I guess gcc will assign cic->key to a register or stack, and it wouldn't be re-readed. Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp> Signed-off-by: Jens Axboe <axboe@suse.de>
-rw-r--r--block/cfq-iosched.c19
1 files changed, 12 insertions, 7 deletions
diff --git a/block/cfq-iosched.c b/block/cfq-iosched.c
index 246feae16c60..2540dfaa3e38 100644
--- a/block/cfq-iosched.c
+++ b/block/cfq-iosched.c
@@ -1487,20 +1487,22 @@ cfq_cic_rb_lookup(struct cfq_data *cfqd, struct io_context *ioc)
1487{ 1487{
1488 struct rb_node *n; 1488 struct rb_node *n;
1489 struct cfq_io_context *cic; 1489 struct cfq_io_context *cic;
1490 void *key = cfqd; 1490 void *k, *key = cfqd;
1491 1491
1492restart: 1492restart:
1493 n = ioc->cic_root.rb_node; 1493 n = ioc->cic_root.rb_node;
1494 while (n) { 1494 while (n) {
1495 cic = rb_entry(n, struct cfq_io_context, rb_node); 1495 cic = rb_entry(n, struct cfq_io_context, rb_node);
1496 if (unlikely(!cic->key)) { 1496 /* ->key must be copied to avoid race with cfq_exit_queue() */
1497 k = cic->key;
1498 if (unlikely(!k)) {
1497 cfq_drop_dead_cic(ioc, cic); 1499 cfq_drop_dead_cic(ioc, cic);
1498 goto restart; 1500 goto restart;
1499 } 1501 }
1500 1502
1501 if (key < cic->key) 1503 if (key < k)
1502 n = n->rb_left; 1504 n = n->rb_left;
1503 else if (key > cic->key) 1505 else if (key > k)
1504 n = n->rb_right; 1506 n = n->rb_right;
1505 else 1507 else
1506 return cic; 1508 return cic;
@@ -1516,6 +1518,7 @@ cfq_cic_link(struct cfq_data *cfqd, struct io_context *ioc,
1516 struct rb_node **p; 1518 struct rb_node **p;
1517 struct rb_node *parent; 1519 struct rb_node *parent;
1518 struct cfq_io_context *__cic; 1520 struct cfq_io_context *__cic;
1521 void *k;
1519 1522
1520 cic->ioc = ioc; 1523 cic->ioc = ioc;
1521 cic->key = cfqd; 1524 cic->key = cfqd;
@@ -1527,14 +1530,16 @@ restart:
1527 while (*p) { 1530 while (*p) {
1528 parent = *p; 1531 parent = *p;
1529 __cic = rb_entry(parent, struct cfq_io_context, rb_node); 1532 __cic = rb_entry(parent, struct cfq_io_context, rb_node);
1530 if (unlikely(!__cic->key)) { 1533 /* ->key must be copied to avoid race with cfq_exit_queue() */
1534 k = __cic->key;
1535 if (unlikely(!k)) {
1531 cfq_drop_dead_cic(ioc, cic); 1536 cfq_drop_dead_cic(ioc, cic);
1532 goto restart; 1537 goto restart;
1533 } 1538 }
1534 1539
1535 if (cic->key < __cic->key) 1540 if (cic->key < k)
1536 p = &(*p)->rb_left; 1541 p = &(*p)->rb_left;
1537 else if (cic->key > __cic->key) 1542 else if (cic->key > k)
1538 p = &(*p)->rb_right; 1543 p = &(*p)->rb_right;
1539 else 1544 else
1540 BUG(); 1545 BUG();