extent-io.c 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * BTRFS filesystem implementation for U-Boot
  4. *
  5. * 2017 Marek Behun, CZ.NIC, marek.behun@nic.cz
  6. */
  7. #include "btrfs.h"
  8. #include <malloc.h>
  9. u64 btrfs_read_extent_inline(struct btrfs_path *path,
  10. struct btrfs_file_extent_item *extent, u64 offset,
  11. u64 size, char *out)
  12. {
  13. u32 clen, dlen, orig_size = size, res;
  14. const char *cbuf;
  15. char *dbuf;
  16. const int data_off = offsetof(struct btrfs_file_extent_item,
  17. disk_bytenr);
  18. clen = btrfs_path_item_size(path) - data_off;
  19. cbuf = (const char *) extent + data_off;
  20. dlen = extent->ram_bytes;
  21. if (offset > dlen)
  22. return -1ULL;
  23. if (size > dlen - offset)
  24. size = dlen - offset;
  25. if (extent->compression == BTRFS_COMPRESS_NONE) {
  26. memcpy(out, cbuf + offset, size);
  27. return size;
  28. }
  29. if (dlen > orig_size) {
  30. dbuf = malloc(dlen);
  31. if (!dbuf)
  32. return -1ULL;
  33. } else {
  34. dbuf = out;
  35. }
  36. res = btrfs_decompress(extent->compression, cbuf, clen, dbuf, dlen);
  37. if (res == -1 || res != dlen)
  38. goto err;
  39. if (dlen > orig_size) {
  40. memcpy(out, dbuf + offset, size);
  41. free(dbuf);
  42. } else if (offset) {
  43. memmove(out, dbuf + offset, size);
  44. }
  45. return size;
  46. err:
  47. if (dlen > orig_size)
  48. free(dbuf);
  49. return -1ULL;
  50. }
  51. u64 btrfs_read_extent_reg(struct btrfs_path *path,
  52. struct btrfs_file_extent_item *extent, u64 offset,
  53. u64 size, char *out)
  54. {
  55. u64 physical, clen, dlen, orig_size = size;
  56. u32 res;
  57. char *cbuf, *dbuf;
  58. clen = extent->disk_num_bytes;
  59. dlen = extent->num_bytes;
  60. if (offset > dlen)
  61. return -1ULL;
  62. if (size > dlen - offset)
  63. size = dlen - offset;
  64. physical = btrfs_map_logical_to_physical(extent->disk_bytenr);
  65. if (physical == -1ULL)
  66. return -1ULL;
  67. if (extent->compression == BTRFS_COMPRESS_NONE) {
  68. physical += extent->offset + offset;
  69. if (!btrfs_devread(physical, size, out))
  70. return -1ULL;
  71. return size;
  72. }
  73. cbuf = malloc(dlen > size ? clen + dlen : clen);
  74. if (!cbuf)
  75. return -1ULL;
  76. if (dlen > orig_size)
  77. dbuf = cbuf + clen;
  78. else
  79. dbuf = out;
  80. if (!btrfs_devread(physical, clen, cbuf))
  81. goto err;
  82. res = btrfs_decompress(extent->compression, cbuf, clen, dbuf, dlen);
  83. if (res == -1)
  84. goto err;
  85. if (dlen > orig_size)
  86. memcpy(out, dbuf + offset, size);
  87. else
  88. memmove(out, dbuf + offset, size);
  89. free(cbuf);
  90. return res;
  91. err:
  92. free(cbuf);
  93. return -1ULL;
  94. }