lguest_blk.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. /* A simple block driver for lguest.
  2. *
  3. * Copyright 2006 Rusty Russell <rusty@rustcorp.com.au> IBM Corporation
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation; either version 2 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program; if not, write to the Free Software
  17. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  18. */
  19. //#define DEBUG
  20. #include <linux/init.h>
  21. #include <linux/types.h>
  22. #include <linux/blkdev.h>
  23. #include <linux/interrupt.h>
  24. #include <linux/lguest_bus.h>
  25. static char next_block_index = 'a';
  26. struct blockdev
  27. {
  28. spinlock_t lock;
  29. /* The disk structure for the kernel. */
  30. struct gendisk *disk;
  31. /* The major number for this disk. */
  32. int major;
  33. int irq;
  34. unsigned long phys_addr;
  35. /* The mapped block page. */
  36. struct lguest_block_page *lb_page;
  37. /* We only have a single request outstanding at a time. */
  38. struct lguest_dma dma;
  39. struct request *req;
  40. };
  41. /* Jens gave me this nice helper to end all chunks of a request. */
  42. static void end_entire_request(struct request *req, int uptodate)
  43. {
  44. if (end_that_request_first(req, uptodate, req->hard_nr_sectors))
  45. BUG();
  46. add_disk_randomness(req->rq_disk);
  47. blkdev_dequeue_request(req);
  48. end_that_request_last(req, uptodate);
  49. }
  50. static irqreturn_t lgb_irq(int irq, void *_bd)
  51. {
  52. struct blockdev *bd = _bd;
  53. unsigned long flags;
  54. if (!bd->req) {
  55. pr_debug("No work!\n");
  56. return IRQ_NONE;
  57. }
  58. if (!bd->lb_page->result) {
  59. pr_debug("No result!\n");
  60. return IRQ_NONE;
  61. }
  62. spin_lock_irqsave(&bd->lock, flags);
  63. end_entire_request(bd->req, bd->lb_page->result == 1);
  64. bd->req = NULL;
  65. bd->dma.used_len = 0;
  66. blk_start_queue(bd->disk->queue);
  67. spin_unlock_irqrestore(&bd->lock, flags);
  68. return IRQ_HANDLED;
  69. }
  70. static unsigned int req_to_dma(struct request *req, struct lguest_dma *dma)
  71. {
  72. unsigned int i = 0, idx, len = 0;
  73. struct bio *bio;
  74. rq_for_each_bio(bio, req) {
  75. struct bio_vec *bvec;
  76. bio_for_each_segment(bvec, bio, idx) {
  77. BUG_ON(i == LGUEST_MAX_DMA_SECTIONS);
  78. BUG_ON(!bvec->bv_len);
  79. dma->addr[i] = page_to_phys(bvec->bv_page)
  80. + bvec->bv_offset;
  81. dma->len[i] = bvec->bv_len;
  82. len += bvec->bv_len;
  83. i++;
  84. }
  85. }
  86. if (i < LGUEST_MAX_DMA_SECTIONS)
  87. dma->len[i] = 0;
  88. return len;
  89. }
  90. static void empty_dma(struct lguest_dma *dma)
  91. {
  92. dma->len[0] = 0;
  93. }
  94. static void setup_req(struct blockdev *bd,
  95. int type, struct request *req, struct lguest_dma *dma)
  96. {
  97. bd->lb_page->type = type;
  98. bd->lb_page->sector = req->sector;
  99. bd->lb_page->result = 0;
  100. bd->req = req;
  101. bd->lb_page->bytes = req_to_dma(req, dma);
  102. }
  103. static void do_write(struct blockdev *bd, struct request *req)
  104. {
  105. struct lguest_dma send;
  106. pr_debug("lgb: WRITE sector %li\n", (long)req->sector);
  107. setup_req(bd, 1, req, &send);
  108. lguest_send_dma(bd->phys_addr, &send);
  109. }
  110. static void do_read(struct blockdev *bd, struct request *req)
  111. {
  112. struct lguest_dma ping;
  113. pr_debug("lgb: READ sector %li\n", (long)req->sector);
  114. setup_req(bd, 0, req, &bd->dma);
  115. empty_dma(&ping);
  116. lguest_send_dma(bd->phys_addr, &ping);
  117. }
  118. static void do_lgb_request(struct request_queue *q)
  119. {
  120. struct blockdev *bd;
  121. struct request *req;
  122. again:
  123. req = elv_next_request(q);
  124. if (!req)
  125. return;
  126. bd = req->rq_disk->private_data;
  127. /* Sometimes we get repeated requests after blk_stop_queue. */
  128. if (bd->req)
  129. return;
  130. if (!blk_fs_request(req)) {
  131. pr_debug("Got non-command 0x%08x\n", req->cmd_type);
  132. req->errors++;
  133. end_entire_request(req, 0);
  134. goto again;
  135. }
  136. if (rq_data_dir(req) == WRITE)
  137. do_write(bd, req);
  138. else
  139. do_read(bd, req);
  140. /* Wait for interrupt to tell us it's done. */
  141. blk_stop_queue(q);
  142. }
  143. static struct block_device_operations lguestblk_fops = {
  144. .owner = THIS_MODULE,
  145. };
  146. static int lguestblk_probe(struct lguest_device *lgdev)
  147. {
  148. struct blockdev *bd;
  149. int err;
  150. int irqflags = IRQF_SHARED;
  151. bd = kmalloc(sizeof(*bd), GFP_KERNEL);
  152. if (!bd)
  153. return -ENOMEM;
  154. spin_lock_init(&bd->lock);
  155. bd->irq = lgdev_irq(lgdev);
  156. bd->req = NULL;
  157. bd->dma.used_len = 0;
  158. bd->dma.len[0] = 0;
  159. bd->phys_addr = (lguest_devices[lgdev->index].pfn << PAGE_SHIFT);
  160. bd->lb_page = lguest_map(bd->phys_addr, 1);
  161. if (!bd->lb_page) {
  162. err = -ENOMEM;
  163. goto out_free_bd;
  164. }
  165. bd->major = register_blkdev(0, "lguestblk");
  166. if (bd->major < 0) {
  167. err = bd->major;
  168. goto out_unmap;
  169. }
  170. bd->disk = alloc_disk(1);
  171. if (!bd->disk) {
  172. err = -ENOMEM;
  173. goto out_unregister_blkdev;
  174. }
  175. bd->disk->queue = blk_init_queue(do_lgb_request, &bd->lock);
  176. if (!bd->disk->queue) {
  177. err = -ENOMEM;
  178. goto out_put_disk;
  179. }
  180. /* We can only handle a certain number of sg entries */
  181. blk_queue_max_hw_segments(bd->disk->queue, LGUEST_MAX_DMA_SECTIONS);
  182. /* Buffers must not cross page boundaries */
  183. blk_queue_segment_boundary(bd->disk->queue, PAGE_SIZE-1);
  184. sprintf(bd->disk->disk_name, "lgb%c", next_block_index++);
  185. if (lguest_devices[lgdev->index].features & LGUEST_DEVICE_F_RANDOMNESS)
  186. irqflags |= IRQF_SAMPLE_RANDOM;
  187. err = request_irq(bd->irq, lgb_irq, irqflags, bd->disk->disk_name, bd);
  188. if (err)
  189. goto out_cleanup_queue;
  190. err = lguest_bind_dma(bd->phys_addr, &bd->dma, 1, bd->irq);
  191. if (err)
  192. goto out_free_irq;
  193. bd->disk->major = bd->major;
  194. bd->disk->first_minor = 0;
  195. bd->disk->private_data = bd;
  196. bd->disk->fops = &lguestblk_fops;
  197. /* This is initialized to the disk size by the other end. */
  198. set_capacity(bd->disk, bd->lb_page->num_sectors);
  199. add_disk(bd->disk);
  200. printk(KERN_INFO "%s: device %i at major %d\n",
  201. bd->disk->disk_name, lgdev->index, bd->major);
  202. lgdev->private = bd;
  203. return 0;
  204. out_free_irq:
  205. free_irq(bd->irq, bd);
  206. out_cleanup_queue:
  207. blk_cleanup_queue(bd->disk->queue);
  208. out_put_disk:
  209. put_disk(bd->disk);
  210. out_unregister_blkdev:
  211. unregister_blkdev(bd->major, "lguestblk");
  212. out_unmap:
  213. lguest_unmap(bd->lb_page);
  214. out_free_bd:
  215. kfree(bd);
  216. return err;
  217. }
  218. static struct lguest_driver lguestblk_drv = {
  219. .name = "lguestblk",
  220. .owner = THIS_MODULE,
  221. .device_type = LGUEST_DEVICE_T_BLOCK,
  222. .probe = lguestblk_probe,
  223. };
  224. static __init int lguestblk_init(void)
  225. {
  226. return register_lguest_driver(&lguestblk_drv);
  227. }
  228. module_init(lguestblk_init);
  229. MODULE_DESCRIPTION("Lguest block driver");
  230. MODULE_LICENSE("GPL");