You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

112 lines
2.5KB

  1. /*
  2. * Copyright (c) 2009-2016 Petri Lehtinen <petri@digip.org>
  3. *
  4. * Jansson is free software; you can redistribute it and/or modify
  5. * it under the terms of the MIT license. See LICENSE for details.
  6. */
  7. #ifndef _GNU_SOURCE
  8. #define _GNU_SOURCE
  9. #endif
  10. #include <stdlib.h>
  11. #include <string.h>
  12. #include "jansson_private.h"
  13. #include "strbuffer.h"
  14. #define STRBUFFER_MIN_SIZE 16
  15. #define STRBUFFER_FACTOR 2
  16. #define STRBUFFER_SIZE_MAX ((size_t)-1)
  17. int strbuffer_init(strbuffer_t *strbuff)
  18. {
  19. strbuff->size = STRBUFFER_MIN_SIZE;
  20. strbuff->length = 0;
  21. strbuff->value = jsonp_malloc(strbuff->size);
  22. if(!strbuff->value)
  23. return -1;
  24. /* initialize to empty */
  25. strbuff->value[0] = '\0';
  26. return 0;
  27. }
  28. void strbuffer_close(strbuffer_t *strbuff)
  29. {
  30. if(strbuff->value)
  31. jsonp_free(strbuff->value);
  32. strbuff->size = 0;
  33. strbuff->length = 0;
  34. strbuff->value = NULL;
  35. }
  36. void strbuffer_clear(strbuffer_t *strbuff)
  37. {
  38. strbuff->length = 0;
  39. strbuff->value[0] = '\0';
  40. }
  41. const char *strbuffer_value(const strbuffer_t *strbuff)
  42. {
  43. return strbuff->value;
  44. }
  45. char *strbuffer_steal_value(strbuffer_t *strbuff)
  46. {
  47. char *result = strbuff->value;
  48. strbuff->value = NULL;
  49. return result;
  50. }
  51. int strbuffer_append_byte(strbuffer_t *strbuff, char byte)
  52. {
  53. return strbuffer_append_bytes(strbuff, &byte, 1);
  54. }
  55. int strbuffer_append_bytes(strbuffer_t *strbuff, const char *data, size_t size)
  56. {
  57. if(size >= strbuff->size - strbuff->length)
  58. {
  59. size_t new_size;
  60. char *new_value;
  61. /* avoid integer overflow */
  62. if (strbuff->size > STRBUFFER_SIZE_MAX / STRBUFFER_FACTOR
  63. || size > STRBUFFER_SIZE_MAX - 1
  64. || strbuff->length > STRBUFFER_SIZE_MAX - 1 - size)
  65. return -1;
  66. new_size = max(strbuff->size * STRBUFFER_FACTOR,
  67. strbuff->length + size + 1);
  68. new_value = jsonp_malloc(new_size);
  69. if(!new_value)
  70. return -1;
  71. memcpy(new_value, strbuff->value, strbuff->length);
  72. jsonp_free(strbuff->value);
  73. strbuff->value = new_value;
  74. strbuff->size = new_size;
  75. }
  76. memcpy(strbuff->value + strbuff->length, data, size);
  77. strbuff->length += size;
  78. strbuff->value[strbuff->length] = '\0';
  79. return 0;
  80. }
  81. char strbuffer_pop(strbuffer_t *strbuff)
  82. {
  83. if(strbuff->length > 0) {
  84. char c = strbuff->value[--strbuff->length];
  85. strbuff->value[strbuff->length] = '\0';
  86. return c;
  87. }
  88. else
  89. return '\0';
  90. }