array-serializer.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. /*
  2. * Copyright (c) 2023 Lain Bailey <[email protected]>
  3. *
  4. * Permission to use, copy, modify, and distribute this software for any
  5. * purpose with or without fee is hereby granted, provided that the above
  6. * copyright notice and this permission notice appear in all copies.
  7. *
  8. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  9. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  10. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  11. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  12. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  13. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  14. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #include "darray.h"
  17. #include "array-serializer.h"
  18. static size_t array_output_write(void *param, const void *data, size_t size)
  19. {
  20. struct array_output_data *output = param;
  21. if (output->cur_pos < output->bytes.num) {
  22. size_t new_size = output->cur_pos + size;
  23. if (new_size > output->bytes.num) {
  24. darray_ensure_capacity(sizeof(uint8_t), &output->bytes.da, new_size);
  25. output->bytes.num = new_size;
  26. }
  27. memcpy(output->bytes.array + output->cur_pos, data, size);
  28. output->cur_pos += size;
  29. } else {
  30. da_push_back_array(output->bytes, (uint8_t *)data, size);
  31. output->cur_pos += size;
  32. }
  33. return size;
  34. }
  35. static int64_t array_output_get_pos(void *param)
  36. {
  37. struct array_output_data *data = param;
  38. return (int64_t)data->bytes.num;
  39. }
  40. static int64_t array_output_seek(void *param, int64_t offset, enum serialize_seek_type seek_type)
  41. {
  42. struct array_output_data *output = param;
  43. size_t new_pos = 0;
  44. switch (seek_type) {
  45. case SERIALIZE_SEEK_START:
  46. new_pos = offset;
  47. break;
  48. case SERIALIZE_SEEK_CURRENT:
  49. new_pos = output->cur_pos + offset;
  50. break;
  51. case SERIALIZE_SEEK_END:
  52. new_pos = output->bytes.num - offset;
  53. break;
  54. }
  55. if (new_pos > output->bytes.num)
  56. return -1;
  57. output->cur_pos = new_pos;
  58. return (int64_t)new_pos;
  59. }
  60. void array_output_serializer_init(struct serializer *s, struct array_output_data *data)
  61. {
  62. memset(s, 0, sizeof(struct serializer));
  63. memset(data, 0, sizeof(struct array_output_data));
  64. s->data = data;
  65. s->write = array_output_write;
  66. s->get_pos = array_output_get_pos;
  67. s->seek = array_output_seek;
  68. }
  69. void array_output_serializer_free(struct array_output_data *data)
  70. {
  71. da_free(data->bytes);
  72. }
  73. void array_output_serializer_reset(struct array_output_data *data)
  74. {
  75. da_clear(data->bytes);
  76. data->cur_pos = 0;
  77. }