array-serializer.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  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),
  25. &output->bytes.da, new_size);
  26. output->bytes.num = new_size;
  27. }
  28. memcpy(output->bytes.array + output->cur_pos, data, size);
  29. output->cur_pos += size;
  30. } else {
  31. da_push_back_array(output->bytes, (uint8_t *)data, size);
  32. output->cur_pos += size;
  33. }
  34. return size;
  35. }
  36. static int64_t array_output_get_pos(void *param)
  37. {
  38. struct array_output_data *data = param;
  39. return (int64_t)data->bytes.num;
  40. }
  41. static int64_t array_output_seek(void *param, int64_t offset,
  42. enum serialize_seek_type seek_type)
  43. {
  44. struct array_output_data *output = param;
  45. size_t new_pos = 0;
  46. switch (seek_type) {
  47. case SERIALIZE_SEEK_START:
  48. new_pos = offset;
  49. break;
  50. case SERIALIZE_SEEK_CURRENT:
  51. new_pos = output->cur_pos + offset;
  52. break;
  53. case SERIALIZE_SEEK_END:
  54. new_pos = output->bytes.num - offset;
  55. break;
  56. }
  57. if (new_pos > output->bytes.num)
  58. return -1;
  59. output->cur_pos = new_pos;
  60. return (int64_t)new_pos;
  61. }
  62. void array_output_serializer_init(struct serializer *s,
  63. struct array_output_data *data)
  64. {
  65. memset(s, 0, sizeof(struct serializer));
  66. memset(data, 0, sizeof(struct array_output_data));
  67. s->data = data;
  68. s->write = array_output_write;
  69. s->get_pos = array_output_get_pos;
  70. s->seek = array_output_seek;
  71. }
  72. void array_output_serializer_free(struct array_output_data *data)
  73. {
  74. da_free(data->bytes);
  75. }
  76. void array_output_serializer_reset(struct array_output_data *data)
  77. {
  78. da_clear(data->bytes);
  79. data->cur_pos = 0;
  80. }