sandbox.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. /*
  2. * Copyright (C) 2013 Henrik Nordstrom <henrik@henriknordstrom.net>
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. */
  6. #include <config.h>
  7. #include <common.h>
  8. #include <part.h>
  9. #include <os.h>
  10. #include <malloc.h>
  11. #include <sandboxblockdev.h>
  12. #include <asm/errno.h>
  13. static struct host_block_dev host_devices[CONFIG_HOST_MAX_DEVICES];
  14. static struct host_block_dev *find_host_device(int dev)
  15. {
  16. if (dev >= 0 && dev < CONFIG_HOST_MAX_DEVICES)
  17. return &host_devices[dev];
  18. return NULL;
  19. }
  20. static unsigned long host_block_read(int dev, unsigned long start,
  21. lbaint_t blkcnt, void *buffer)
  22. {
  23. struct host_block_dev *host_dev = find_host_device(dev);
  24. if (!host_dev)
  25. return -1;
  26. if (os_lseek(host_dev->fd,
  27. start * host_dev->blk_dev.blksz,
  28. OS_SEEK_SET) == -1) {
  29. printf("ERROR: Invalid position\n");
  30. return -1;
  31. }
  32. ssize_t len = os_read(host_dev->fd, buffer,
  33. blkcnt * host_dev->blk_dev.blksz);
  34. if (len >= 0)
  35. return len / host_dev->blk_dev.blksz;
  36. return -1;
  37. }
  38. static unsigned long host_block_write(int dev, unsigned long start,
  39. lbaint_t blkcnt, const void *buffer)
  40. {
  41. struct host_block_dev *host_dev = find_host_device(dev);
  42. if (os_lseek(host_dev->fd,
  43. start * host_dev->blk_dev.blksz,
  44. OS_SEEK_SET) == -1) {
  45. printf("ERROR: Invalid position\n");
  46. return -1;
  47. }
  48. ssize_t len = os_write(host_dev->fd, buffer, blkcnt *
  49. host_dev->blk_dev.blksz);
  50. if (len >= 0)
  51. return len / host_dev->blk_dev.blksz;
  52. return -1;
  53. }
  54. int host_dev_bind(int dev, char *filename)
  55. {
  56. struct host_block_dev *host_dev = find_host_device(dev);
  57. if (!host_dev)
  58. return -1;
  59. if (host_dev->blk_dev.priv) {
  60. os_close(host_dev->fd);
  61. host_dev->blk_dev.priv = NULL;
  62. }
  63. if (host_dev->filename)
  64. free(host_dev->filename);
  65. if (filename && *filename) {
  66. host_dev->filename = strdup(filename);
  67. } else {
  68. host_dev->filename = NULL;
  69. return 0;
  70. }
  71. host_dev->fd = os_open(host_dev->filename, OS_O_RDWR);
  72. if (host_dev->fd == -1) {
  73. printf("Failed to access host backing file '%s'\n",
  74. host_dev->filename);
  75. return 1;
  76. }
  77. block_dev_desc_t *blk_dev = &host_dev->blk_dev;
  78. blk_dev->if_type = IF_TYPE_HOST;
  79. blk_dev->priv = host_dev;
  80. blk_dev->blksz = 512;
  81. blk_dev->lba = os_lseek(host_dev->fd, 0, OS_SEEK_END) / blk_dev->blksz;
  82. blk_dev->block_read = host_block_read;
  83. blk_dev->block_write = host_block_write;
  84. blk_dev->dev = dev;
  85. blk_dev->part_type = PART_TYPE_UNKNOWN;
  86. init_part(blk_dev);
  87. return 0;
  88. }
  89. int host_get_dev_err(int dev, block_dev_desc_t **blk_devp)
  90. {
  91. struct host_block_dev *host_dev = find_host_device(dev);
  92. if (!host_dev)
  93. return -ENODEV;
  94. if (!host_dev->blk_dev.priv)
  95. return -ENOENT;
  96. *blk_devp = &host_dev->blk_dev;
  97. return 0;
  98. }
  99. block_dev_desc_t *host_get_dev(int dev)
  100. {
  101. block_dev_desc_t *blk_dev;
  102. if (host_get_dev_err(dev, &blk_dev))
  103. return NULL;
  104. return blk_dev;
  105. }