mtd_blkdevs.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482
  1. /*
  2. * $Id: mtd_blkdevs.c,v 1.26 2005/07/29 19:42:04 tpoynor Exp $
  3. *
  4. * (C) 2003 David Woodhouse <dwmw2@infradead.org>
  5. *
  6. * Interface to Linux 2.5 block layer for MTD 'translation layers'.
  7. *
  8. */
  9. #include <linux/kernel.h>
  10. #include <linux/slab.h>
  11. #include <linux/module.h>
  12. #include <linux/list.h>
  13. #include <linux/fs.h>
  14. #include <linux/mtd/blktrans.h>
  15. #include <linux/mtd/mtd.h>
  16. #include <linux/blkdev.h>
  17. #include <linux/blkpg.h>
  18. #include <linux/spinlock.h>
  19. #include <linux/hdreg.h>
  20. #include <linux/init.h>
  21. #include <asm/semaphore.h>
  22. #include <asm/uaccess.h>
  23. static LIST_HEAD(blktrans_majors);
  24. extern struct semaphore mtd_table_mutex;
  25. extern struct mtd_info *mtd_table[];
  26. struct mtd_blkcore_priv {
  27. struct completion thread_dead;
  28. int exiting;
  29. wait_queue_head_t thread_wq;
  30. struct request_queue *rq;
  31. spinlock_t queue_lock;
  32. };
  33. static int do_blktrans_request(struct mtd_blktrans_ops *tr,
  34. struct mtd_blktrans_dev *dev,
  35. struct request *req)
  36. {
  37. unsigned long block, nsect;
  38. char *buf;
  39. block = req->sector;
  40. nsect = req->current_nr_sectors;
  41. buf = req->buffer;
  42. if (!(req->flags & REQ_CMD))
  43. return 0;
  44. if (block + nsect > get_capacity(req->rq_disk))
  45. return 0;
  46. switch(rq_data_dir(req)) {
  47. case READ:
  48. for (; nsect > 0; nsect--, block++, buf += 512)
  49. if (tr->readsect(dev, block, buf))
  50. return 0;
  51. return 1;
  52. case WRITE:
  53. if (!tr->writesect)
  54. return 0;
  55. for (; nsect > 0; nsect--, block++, buf += 512)
  56. if (tr->writesect(dev, block, buf))
  57. return 0;
  58. return 1;
  59. default:
  60. printk(KERN_NOTICE "Unknown request %ld\n", rq_data_dir(req));
  61. return 0;
  62. }
  63. }
  64. static int mtd_blktrans_thread(void *arg)
  65. {
  66. struct mtd_blktrans_ops *tr = arg;
  67. struct request_queue *rq = tr->blkcore_priv->rq;
  68. /* we might get involved when memory gets low, so use PF_MEMALLOC */
  69. current->flags |= PF_MEMALLOC | PF_NOFREEZE;
  70. daemonize("%sd", tr->name);
  71. /* daemonize() doesn't do this for us since some kernel threads
  72. actually want to deal with signals. We can't just call
  73. exit_sighand() since that'll cause an oops when we finally
  74. do exit. */
  75. spin_lock_irq(&current->sighand->siglock);
  76. sigfillset(&current->blocked);
  77. recalc_sigpending();
  78. spin_unlock_irq(&current->sighand->siglock);
  79. spin_lock_irq(rq->queue_lock);
  80. while (!tr->blkcore_priv->exiting) {
  81. struct request *req;
  82. struct mtd_blktrans_dev *dev;
  83. int res = 0;
  84. DECLARE_WAITQUEUE(wait, current);
  85. req = elv_next_request(rq);
  86. if (!req) {
  87. add_wait_queue(&tr->blkcore_priv->thread_wq, &wait);
  88. set_current_state(TASK_INTERRUPTIBLE);
  89. spin_unlock_irq(rq->queue_lock);
  90. schedule();
  91. remove_wait_queue(&tr->blkcore_priv->thread_wq, &wait);
  92. spin_lock_irq(rq->queue_lock);
  93. continue;
  94. }
  95. dev = req->rq_disk->private_data;
  96. tr = dev->tr;
  97. spin_unlock_irq(rq->queue_lock);
  98. down(&dev->sem);
  99. res = do_blktrans_request(tr, dev, req);
  100. up(&dev->sem);
  101. spin_lock_irq(rq->queue_lock);
  102. end_request(req, res);
  103. }
  104. spin_unlock_irq(rq->queue_lock);
  105. complete_and_exit(&tr->blkcore_priv->thread_dead, 0);
  106. }
  107. static void mtd_blktrans_request(struct request_queue *rq)
  108. {
  109. struct mtd_blktrans_ops *tr = rq->queuedata;
  110. wake_up(&tr->blkcore_priv->thread_wq);
  111. }
  112. static int blktrans_open(struct inode *i, struct file *f)
  113. {
  114. struct mtd_blktrans_dev *dev;
  115. struct mtd_blktrans_ops *tr;
  116. int ret = -ENODEV;
  117. dev = i->i_bdev->bd_disk->private_data;
  118. tr = dev->tr;
  119. if (!try_module_get(dev->mtd->owner))
  120. goto out;
  121. if (!try_module_get(tr->owner))
  122. goto out_tr;
  123. /* FIXME: Locking. A hot pluggable device can go away
  124. (del_mtd_device can be called for it) without its module
  125. being unloaded. */
  126. dev->mtd->usecount++;
  127. ret = 0;
  128. if (tr->open && (ret = tr->open(dev))) {
  129. dev->mtd->usecount--;
  130. module_put(dev->mtd->owner);
  131. out_tr:
  132. module_put(tr->owner);
  133. }
  134. out:
  135. return ret;
  136. }
  137. static int blktrans_release(struct inode *i, struct file *f)
  138. {
  139. struct mtd_blktrans_dev *dev;
  140. struct mtd_blktrans_ops *tr;
  141. int ret = 0;
  142. dev = i->i_bdev->bd_disk->private_data;
  143. tr = dev->tr;
  144. if (tr->release)
  145. ret = tr->release(dev);
  146. if (!ret) {
  147. dev->mtd->usecount--;
  148. module_put(dev->mtd->owner);
  149. module_put(tr->owner);
  150. }
  151. return ret;
  152. }
  153. static int blktrans_ioctl(struct inode *inode, struct file *file,
  154. unsigned int cmd, unsigned long arg)
  155. {
  156. struct mtd_blktrans_dev *dev = inode->i_bdev->bd_disk->private_data;
  157. struct mtd_blktrans_ops *tr = dev->tr;
  158. switch (cmd) {
  159. case BLKFLSBUF:
  160. if (tr->flush)
  161. return tr->flush(dev);
  162. /* The core code did the work, we had nothing to do. */
  163. return 0;
  164. case HDIO_GETGEO:
  165. if (tr->getgeo) {
  166. struct hd_geometry g;
  167. int ret;
  168. memset(&g, 0, sizeof(g));
  169. ret = tr->getgeo(dev, &g);
  170. if (ret)
  171. return ret;
  172. g.start = get_start_sect(inode->i_bdev);
  173. if (copy_to_user((void __user *)arg, &g, sizeof(g)))
  174. return -EFAULT;
  175. return 0;
  176. } /* else */
  177. default:
  178. return -ENOTTY;
  179. }
  180. }
  181. struct block_device_operations mtd_blktrans_ops = {
  182. .owner = THIS_MODULE,
  183. .open = blktrans_open,
  184. .release = blktrans_release,
  185. .ioctl = blktrans_ioctl,
  186. };
  187. int add_mtd_blktrans_dev(struct mtd_blktrans_dev *new)
  188. {
  189. struct mtd_blktrans_ops *tr = new->tr;
  190. struct list_head *this;
  191. int last_devnum = -1;
  192. struct gendisk *gd;
  193. if (!down_trylock(&mtd_table_mutex)) {
  194. up(&mtd_table_mutex);
  195. BUG();
  196. }
  197. list_for_each(this, &tr->devs) {
  198. struct mtd_blktrans_dev *d = list_entry(this, struct mtd_blktrans_dev, list);
  199. if (new->devnum == -1) {
  200. /* Use first free number */
  201. if (d->devnum != last_devnum+1) {
  202. /* Found a free devnum. Plug it in here */
  203. new->devnum = last_devnum+1;
  204. list_add_tail(&new->list, &d->list);
  205. goto added;
  206. }
  207. } else if (d->devnum == new->devnum) {
  208. /* Required number taken */
  209. return -EBUSY;
  210. } else if (d->devnum > new->devnum) {
  211. /* Required number was free */
  212. list_add_tail(&new->list, &d->list);
  213. goto added;
  214. }
  215. last_devnum = d->devnum;
  216. }
  217. if (new->devnum == -1)
  218. new->devnum = last_devnum+1;
  219. if ((new->devnum << tr->part_bits) > 256) {
  220. return -EBUSY;
  221. }
  222. init_MUTEX(&new->sem);
  223. list_add_tail(&new->list, &tr->devs);
  224. added:
  225. if (!tr->writesect)
  226. new->readonly = 1;
  227. gd = alloc_disk(1 << tr->part_bits);
  228. if (!gd) {
  229. list_del(&new->list);
  230. return -ENOMEM;
  231. }
  232. gd->major = tr->major;
  233. gd->first_minor = (new->devnum) << tr->part_bits;
  234. gd->fops = &mtd_blktrans_ops;
  235. if (tr->part_bits)
  236. if (new->devnum < 26)
  237. snprintf(gd->disk_name, sizeof(gd->disk_name),
  238. "%s%c", tr->name, 'a' + new->devnum);
  239. else
  240. snprintf(gd->disk_name, sizeof(gd->disk_name),
  241. "%s%c%c", tr->name,
  242. 'a' - 1 + new->devnum / 26,
  243. 'a' + new->devnum % 26);
  244. else
  245. snprintf(gd->disk_name, sizeof(gd->disk_name),
  246. "%s%d", tr->name, new->devnum);
  247. /* 2.5 has capacity in units of 512 bytes while still
  248. having BLOCK_SIZE_BITS set to 10. Just to keep us amused. */
  249. set_capacity(gd, (new->size * new->blksize) >> 9);
  250. gd->private_data = new;
  251. new->blkcore_priv = gd;
  252. gd->queue = tr->blkcore_priv->rq;
  253. if (new->readonly)
  254. set_disk_ro(gd, 1);
  255. add_disk(gd);
  256. return 0;
  257. }
  258. int del_mtd_blktrans_dev(struct mtd_blktrans_dev *old)
  259. {
  260. if (!down_trylock(&mtd_table_mutex)) {
  261. up(&mtd_table_mutex);
  262. BUG();
  263. }
  264. list_del(&old->list);
  265. del_gendisk(old->blkcore_priv);
  266. put_disk(old->blkcore_priv);
  267. return 0;
  268. }
  269. static void blktrans_notify_remove(struct mtd_info *mtd)
  270. {
  271. struct list_head *this, *this2, *next;
  272. list_for_each(this, &blktrans_majors) {
  273. struct mtd_blktrans_ops *tr = list_entry(this, struct mtd_blktrans_ops, list);
  274. list_for_each_safe(this2, next, &tr->devs) {
  275. struct mtd_blktrans_dev *dev = list_entry(this2, struct mtd_blktrans_dev, list);
  276. if (dev->mtd == mtd)
  277. tr->remove_dev(dev);
  278. }
  279. }
  280. }
  281. static void blktrans_notify_add(struct mtd_info *mtd)
  282. {
  283. struct list_head *this;
  284. if (mtd->type == MTD_ABSENT)
  285. return;
  286. list_for_each(this, &blktrans_majors) {
  287. struct mtd_blktrans_ops *tr = list_entry(this, struct mtd_blktrans_ops, list);
  288. tr->add_mtd(tr, mtd);
  289. }
  290. }
  291. static struct mtd_notifier blktrans_notifier = {
  292. .add = blktrans_notify_add,
  293. .remove = blktrans_notify_remove,
  294. };
  295. int register_mtd_blktrans(struct mtd_blktrans_ops *tr)
  296. {
  297. int ret, i;
  298. /* Register the notifier if/when the first device type is
  299. registered, to prevent the link/init ordering from fucking
  300. us over. */
  301. if (!blktrans_notifier.list.next)
  302. register_mtd_user(&blktrans_notifier);
  303. tr->blkcore_priv = kmalloc(sizeof(*tr->blkcore_priv), GFP_KERNEL);
  304. if (!tr->blkcore_priv)
  305. return -ENOMEM;
  306. memset(tr->blkcore_priv, 0, sizeof(*tr->blkcore_priv));
  307. down(&mtd_table_mutex);
  308. ret = register_blkdev(tr->major, tr->name);
  309. if (ret) {
  310. printk(KERN_WARNING "Unable to register %s block device on major %d: %d\n",
  311. tr->name, tr->major, ret);
  312. kfree(tr->blkcore_priv);
  313. up(&mtd_table_mutex);
  314. return ret;
  315. }
  316. spin_lock_init(&tr->blkcore_priv->queue_lock);
  317. init_completion(&tr->blkcore_priv->thread_dead);
  318. init_waitqueue_head(&tr->blkcore_priv->thread_wq);
  319. tr->blkcore_priv->rq = blk_init_queue(mtd_blktrans_request, &tr->blkcore_priv->queue_lock);
  320. if (!tr->blkcore_priv->rq) {
  321. unregister_blkdev(tr->major, tr->name);
  322. kfree(tr->blkcore_priv);
  323. up(&mtd_table_mutex);
  324. return -ENOMEM;
  325. }
  326. tr->blkcore_priv->rq->queuedata = tr;
  327. ret = kernel_thread(mtd_blktrans_thread, tr, CLONE_KERNEL);
  328. if (ret < 0) {
  329. blk_cleanup_queue(tr->blkcore_priv->rq);
  330. unregister_blkdev(tr->major, tr->name);
  331. kfree(tr->blkcore_priv);
  332. up(&mtd_table_mutex);
  333. return ret;
  334. }
  335. INIT_LIST_HEAD(&tr->devs);
  336. list_add(&tr->list, &blktrans_majors);
  337. for (i=0; i<MAX_MTD_DEVICES; i++) {
  338. if (mtd_table[i] && mtd_table[i]->type != MTD_ABSENT)
  339. tr->add_mtd(tr, mtd_table[i]);
  340. }
  341. up(&mtd_table_mutex);
  342. return 0;
  343. }
  344. int deregister_mtd_blktrans(struct mtd_blktrans_ops *tr)
  345. {
  346. struct list_head *this, *next;
  347. down(&mtd_table_mutex);
  348. /* Clean up the kernel thread */
  349. tr->blkcore_priv->exiting = 1;
  350. wake_up(&tr->blkcore_priv->thread_wq);
  351. wait_for_completion(&tr->blkcore_priv->thread_dead);
  352. /* Remove it from the list of active majors */
  353. list_del(&tr->list);
  354. list_for_each_safe(this, next, &tr->devs) {
  355. struct mtd_blktrans_dev *dev = list_entry(this, struct mtd_blktrans_dev, list);
  356. tr->remove_dev(dev);
  357. }
  358. blk_cleanup_queue(tr->blkcore_priv->rq);
  359. unregister_blkdev(tr->major, tr->name);
  360. up(&mtd_table_mutex);
  361. kfree(tr->blkcore_priv);
  362. if (!list_empty(&tr->devs))
  363. BUG();
  364. return 0;
  365. }
  366. static void __exit mtd_blktrans_exit(void)
  367. {
  368. /* No race here -- if someone's currently in register_mtd_blktrans
  369. we're screwed anyway. */
  370. if (blktrans_notifier.list.next)
  371. unregister_mtd_user(&blktrans_notifier);
  372. }
  373. module_exit(mtd_blktrans_exit);
  374. EXPORT_SYMBOL_GPL(register_mtd_blktrans);
  375. EXPORT_SYMBOL_GPL(deregister_mtd_blktrans);
  376. EXPORT_SYMBOL_GPL(add_mtd_blktrans_dev);
  377. EXPORT_SYMBOL_GPL(del_mtd_blktrans_dev);
  378. MODULE_AUTHOR("David Woodhouse <dwmw2@infradead.org>");
  379. MODULE_LICENSE("GPL");
  380. MODULE_DESCRIPTION("Common interface to block layer for MTD 'translation layers'");