arithmetic_storage_io.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. /*
  2. * Copyright (c) 2020, Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <errno.h>
  7. #include <stream.h>
  8. #include <mem_manager.h>
  9. #include <arithmetic_storage_io.h>
  10. #define STORAGE_BLOCK (2048)
  11. #define STORAGE_BLOCK_MASK (0xFFFFF800)
  12. #ifdef CONFIG_ACTIONS_PARSER
  13. static u8_t storage_block_buff[STORAGE_BLOCK];
  14. #endif
  15. struct storage_block_info_t {
  16. storage_io_t *io;
  17. int data_size;
  18. int file_offset;
  19. int r_offset;
  20. u8_t *buff;
  21. };
  22. static struct storage_block_info_t storage_block = {
  23. .io = NULL,
  24. };
  25. static int _storage_stream_read(void *buf, int size, int count, storage_io_t *io)
  26. {
  27. struct storage_block_info_t *info = &storage_block;
  28. if (!io || !io->hnd)
  29. return -EINVAL;
  30. if (size != 1)
  31. count *= size;
  32. if (info->io && info->io == io) {
  33. io_stream_t stream = (io_stream_t)io->hnd;
  34. size = count;
  35. while((size > 0) && (info->r_offset < stream->total_size)) {
  36. if((info->r_offset - info->file_offset) >= info->data_size) {
  37. info->file_offset += info->data_size;
  38. info->data_size = stream_read(stream, info->buff, STORAGE_BLOCK);
  39. if(info->data_size <= 0) {
  40. if(size == count) {
  41. return info->data_size;
  42. } else {
  43. break;
  44. }
  45. }
  46. continue;
  47. }
  48. int len = info->file_offset + info->data_size - info->r_offset;
  49. len = min(len, size);
  50. memcpy(buf, &info->buff[info->r_offset - info->file_offset], len);
  51. info->r_offset += len;
  52. size -= len;
  53. buf = (char*)buf + len;
  54. }
  55. return count - size;
  56. } else {
  57. int len = stream_read((io_stream_t)io->hnd, buf, count);
  58. return len;
  59. }
  60. }
  61. static int _storage_stream_write(void *buf, int size, int count, storage_io_t *io)
  62. {
  63. if (!io || !io->hnd)
  64. return -EINVAL;
  65. if (size != 1)
  66. count *= size;
  67. return stream_write((io_stream_t)io->hnd, buf, count);
  68. }
  69. static int _storage_stream_seek(storage_io_t *io, int offset, int whence)
  70. {
  71. struct storage_block_info_t *info = &storage_block;
  72. if (!io || !io->hnd)
  73. return -EINVAL;
  74. if (info->io && info->io == io) {
  75. io_stream_t stream = (io_stream_t)io->hnd;
  76. int target_off;
  77. switch(whence) {
  78. case SEEK_DIR_BEG:
  79. target_off = offset;
  80. break;
  81. case SEEK_DIR_CUR:
  82. target_off = info->r_offset + offset;
  83. break;
  84. case SEEK_DIR_END:
  85. target_off = stream->total_size + offset;
  86. break;
  87. default:
  88. SYS_LOG_ERR("mode not support 0x%x \n", whence);
  89. return -1;
  90. }
  91. if(target_off < 0) {
  92. SYS_LOG_WRN("err offset %d, %d, %d\n", offset, whence, target_off);
  93. target_off = 0;
  94. } else if(target_off > stream->total_size) {
  95. SYS_LOG_WRN("err offset %d, %d, %d\n", offset, whence, target_off);
  96. target_off = stream->total_size;
  97. }
  98. info->data_size = 0;
  99. info->file_offset = target_off & STORAGE_BLOCK_MASK;
  100. info->r_offset = target_off;
  101. return stream_seek((io_stream_t)io->hnd, info->file_offset, SEEK_DIR_BEG);
  102. } else {
  103. /* STORAGEIO_SEEK_xxx equal to io_stream_t SEEK_DIR_xxx */
  104. return stream_seek((io_stream_t)io->hnd, offset, whence);
  105. }
  106. }
  107. static int _storage_stream_tell(storage_io_t *io, int mode)
  108. {
  109. io_stream_t stream;
  110. if (!io || !io->hnd)
  111. return -EINVAL;
  112. stream = (io_stream_t)io->hnd;
  113. switch (mode) {
  114. case STORAGEIO_TELL_CUR:
  115. if(storage_block.io && (storage_block.io == io)) {
  116. return storage_block.r_offset;
  117. } else {
  118. return stream_tell(stream);
  119. }
  120. break;
  121. case STORAGEIO_TELL_END:
  122. return stream->total_size;
  123. default:
  124. return -EINVAL;
  125. }
  126. }
  127. storage_io_t *storage_io_block_stream(void *stream)
  128. {
  129. storage_io_t *io = mem_malloc(sizeof(storage_io_t));
  130. if (io) {
  131. io->hnd = stream;
  132. io->read = _storage_stream_read;
  133. io->write = _storage_stream_write;
  134. io->seek = _storage_stream_seek;
  135. io->tell = _storage_stream_tell;
  136. }
  137. struct storage_block_info_t *info = &storage_block;
  138. if (!info->io) {
  139. memset(info, 0 , sizeof(struct storage_block_info_t));
  140. #ifdef CONFIG_ACTIONS_PARSER
  141. info->buff = storage_block_buff;
  142. #endif
  143. info->io = io;
  144. }
  145. return io;
  146. }
  147. storage_io_t *storage_io_wrap_stream(void *stream)
  148. {
  149. storage_io_t *io = mem_malloc(sizeof(storage_io_t));
  150. if (io) {
  151. io->hnd = stream;
  152. io->read = _storage_stream_read;
  153. io->write = _storage_stream_write;
  154. io->seek = _storage_stream_seek;
  155. io->tell = _storage_stream_tell;
  156. }
  157. return io;
  158. }
  159. void storage_io_unwrap_stream(storage_io_t *io)
  160. {
  161. struct storage_block_info_t *info = &storage_block;
  162. if (info->io && info->io == io) {
  163. memset(info, 0 , sizeof(struct storage_block_info_t));
  164. info->io = NULL;
  165. }
  166. mem_free(io);
  167. }