sandbox.c 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2013 Henrik Nordstrom <henrik@henriknordstrom.net>
  4. */
  5. #include <common.h>
  6. #include <blk.h>
  7. #include <dm.h>
  8. #include <fdtdec.h>
  9. #include <part.h>
  10. #include <os.h>
  11. #include <malloc.h>
  12. #include <sandboxblockdev.h>
  13. #include <linux/errno.h>
  14. #include <dm/device-internal.h>
  15. DECLARE_GLOBAL_DATA_PTR;
  16. #ifndef CONFIG_BLK
  17. static struct host_block_dev host_devices[CONFIG_HOST_MAX_DEVICES];
  18. static struct host_block_dev *find_host_device(int dev)
  19. {
  20. if (dev >= 0 && dev < CONFIG_HOST_MAX_DEVICES)
  21. return &host_devices[dev];
  22. return NULL;
  23. }
  24. #endif
  25. #ifdef CONFIG_BLK
  26. static unsigned long host_block_read(struct udevice *dev,
  27. unsigned long start, lbaint_t blkcnt,
  28. void *buffer)
  29. {
  30. struct host_block_dev *host_dev = dev_get_priv(dev);
  31. struct blk_desc *block_dev = dev_get_uclass_platdata(dev);
  32. #else
  33. static unsigned long host_block_read(struct blk_desc *block_dev,
  34. unsigned long start, lbaint_t blkcnt,
  35. void *buffer)
  36. {
  37. int dev = block_dev->devnum;
  38. struct host_block_dev *host_dev = find_host_device(dev);
  39. if (!host_dev)
  40. return -1;
  41. #endif
  42. if (os_lseek(host_dev->fd, start * block_dev->blksz, OS_SEEK_SET) ==
  43. -1) {
  44. printf("ERROR: Invalid block %lx\n", start);
  45. return -1;
  46. }
  47. ssize_t len = os_read(host_dev->fd, buffer, blkcnt * block_dev->blksz);
  48. if (len >= 0)
  49. return len / block_dev->blksz;
  50. return -1;
  51. }
  52. #ifdef CONFIG_BLK
  53. static unsigned long host_block_write(struct udevice *dev,
  54. unsigned long start, lbaint_t blkcnt,
  55. const void *buffer)
  56. {
  57. struct host_block_dev *host_dev = dev_get_priv(dev);
  58. struct blk_desc *block_dev = dev_get_uclass_platdata(dev);
  59. #else
  60. static unsigned long host_block_write(struct blk_desc *block_dev,
  61. unsigned long start, lbaint_t blkcnt,
  62. const void *buffer)
  63. {
  64. int dev = block_dev->devnum;
  65. struct host_block_dev *host_dev = find_host_device(dev);
  66. #endif
  67. if (os_lseek(host_dev->fd, start * block_dev->blksz, OS_SEEK_SET) ==
  68. -1) {
  69. printf("ERROR: Invalid block %lx\n", start);
  70. return -1;
  71. }
  72. ssize_t len = os_write(host_dev->fd, buffer, blkcnt * block_dev->blksz);
  73. if (len >= 0)
  74. return len / block_dev->blksz;
  75. return -1;
  76. }
  77. #ifdef CONFIG_BLK
  78. int host_dev_bind(int devnum, char *filename)
  79. {
  80. struct host_block_dev *host_dev;
  81. struct udevice *dev;
  82. char dev_name[20], *str, *fname;
  83. int ret, fd;
  84. /* Remove and unbind the old device, if any */
  85. ret = blk_get_device(IF_TYPE_HOST, devnum, &dev);
  86. if (ret == 0) {
  87. ret = device_remove(dev, DM_REMOVE_NORMAL);
  88. if (ret)
  89. return ret;
  90. ret = device_unbind(dev);
  91. if (ret)
  92. return ret;
  93. } else if (ret != -ENODEV) {
  94. return ret;
  95. }
  96. if (!filename)
  97. return 0;
  98. snprintf(dev_name, sizeof(dev_name), "host%d", devnum);
  99. str = strdup(dev_name);
  100. if (!str)
  101. return -ENOMEM;
  102. fname = strdup(filename);
  103. if (!fname) {
  104. free(str);
  105. return -ENOMEM;
  106. }
  107. fd = os_open(filename, OS_O_RDWR);
  108. if (fd == -1) {
  109. printf("Failed to access host backing file '%s'\n", filename);
  110. ret = -ENOENT;
  111. goto err;
  112. }
  113. ret = blk_create_device(gd->dm_root, "sandbox_host_blk", str,
  114. IF_TYPE_HOST, devnum, 512,
  115. os_lseek(fd, 0, OS_SEEK_END) / 512, &dev);
  116. if (ret)
  117. goto err_file;
  118. ret = device_probe(dev);
  119. if (ret) {
  120. device_unbind(dev);
  121. goto err_file;
  122. }
  123. host_dev = dev_get_priv(dev);
  124. host_dev->fd = fd;
  125. host_dev->filename = fname;
  126. return blk_prepare_device(dev);
  127. err_file:
  128. os_close(fd);
  129. err:
  130. free(fname);
  131. free(str);
  132. return ret;
  133. }
  134. #else
  135. int host_dev_bind(int dev, char *filename)
  136. {
  137. struct host_block_dev *host_dev = find_host_device(dev);
  138. if (!host_dev)
  139. return -1;
  140. if (host_dev->blk_dev.priv) {
  141. os_close(host_dev->fd);
  142. host_dev->blk_dev.priv = NULL;
  143. }
  144. if (host_dev->filename)
  145. free(host_dev->filename);
  146. if (filename && *filename) {
  147. host_dev->filename = strdup(filename);
  148. } else {
  149. host_dev->filename = NULL;
  150. return 0;
  151. }
  152. host_dev->fd = os_open(host_dev->filename, OS_O_RDWR);
  153. if (host_dev->fd == -1) {
  154. printf("Failed to access host backing file '%s'\n",
  155. host_dev->filename);
  156. return 1;
  157. }
  158. struct blk_desc *blk_dev = &host_dev->blk_dev;
  159. blk_dev->if_type = IF_TYPE_HOST;
  160. blk_dev->priv = host_dev;
  161. blk_dev->blksz = 512;
  162. blk_dev->lba = os_lseek(host_dev->fd, 0, OS_SEEK_END) / blk_dev->blksz;
  163. blk_dev->block_read = host_block_read;
  164. blk_dev->block_write = host_block_write;
  165. blk_dev->devnum = dev;
  166. blk_dev->part_type = PART_TYPE_UNKNOWN;
  167. part_init(blk_dev);
  168. return 0;
  169. }
  170. #endif
  171. int host_get_dev_err(int devnum, struct blk_desc **blk_devp)
  172. {
  173. #ifdef CONFIG_BLK
  174. struct udevice *dev;
  175. int ret;
  176. ret = blk_get_device(IF_TYPE_HOST, devnum, &dev);
  177. if (ret)
  178. return ret;
  179. *blk_devp = dev_get_uclass_platdata(dev);
  180. #else
  181. struct host_block_dev *host_dev = find_host_device(devnum);
  182. if (!host_dev)
  183. return -ENODEV;
  184. if (!host_dev->blk_dev.priv)
  185. return -ENOENT;
  186. *blk_devp = &host_dev->blk_dev;
  187. #endif
  188. return 0;
  189. }
  190. #ifdef CONFIG_BLK
  191. static const struct blk_ops sandbox_host_blk_ops = {
  192. .read = host_block_read,
  193. .write = host_block_write,
  194. };
  195. U_BOOT_DRIVER(sandbox_host_blk) = {
  196. .name = "sandbox_host_blk",
  197. .id = UCLASS_BLK,
  198. .ops = &sandbox_host_blk_ops,
  199. .priv_auto_alloc_size = sizeof(struct host_block_dev),
  200. };
  201. #else
  202. U_BOOT_LEGACY_BLK(sandbox_host) = {
  203. .if_typename = "host",
  204. .if_type = IF_TYPE_HOST,
  205. .max_devs = CONFIG_HOST_MAX_DEVICES,
  206. .get_dev = host_get_dev_err,
  207. };
  208. #endif