buffer.h 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #ifndef _LSUP_BUFFER_H
  2. #define _LSUP_BUFFER_H
  3. #include "xxhash.h"
  4. #include "core.h"
  5. #ifndef HASH_SEED
  6. /** @brief Seed used for all hashing. Compile-time configurable.
  7. */
  8. #define HASH_SEED 0
  9. #endif
  10. // "NULL" key, a value that is never user-provided. Used to mark special
  11. // values (e.g. deleted records).
  12. #define NULL_KEY 0
  13. /** @brief General-purpose data buffer.
  14. *
  15. * The structure is transparently exposed so that the related API only defines
  16. * few basic helper methods. Other operations, such as appending, may be
  17. * performed by simply using the addr and size attributes.
  18. *
  19. * A buffer can be initialized once and reused multiple times, e.g. in a loop,
  20. * without being freed between iterations, by using #LSUP_buffer_init.
  21. */
  22. typedef struct LSUP_Buffer {
  23. /*@null@*/ void *addr;
  24. size_t size;
  25. } LSUP_Buffer;
  26. /** Initialize or reuse a buffer handle.
  27. *
  28. * The handle must have been created with #LSUP_buffer_new*().
  29. *
  30. * The data block is resized without being freed first. The handle must be
  31. * eventually freed with #LSUP_buffer_done() after use.
  32. *
  33. * @param buf[in] A buffer handle obtained with #LSUP_buffer_new or by manual
  34. * allocation.
  35. *
  36. * @param size[in] New size.
  37. *
  38. * @param data[in] If not NULL, data to replace the existing ones. The size
  39. * of the data to be copied is determined by the size parameter. If NULL, the
  40. * existing data are preserved as with a normal realloc().
  41. */
  42. LSUP_rc
  43. LSUP_buffer_init (LSUP_Buffer *buf, const size_t size, const void *data);
  44. /** @brief Create a new buffer and optionally populate it with data.
  45. *
  46. * To change the buffer size and/or data later call #LSUP_buffer_init.
  47. *
  48. * To copy a buffer just do buf2 = LSUP_buffer_new (buf1->size, buf1->addr);
  49. *
  50. * @param size[in] Length of the data.
  51. *
  52. * @param data[in] Optional data to initially populate the object with. If
  53. * NULL, the buffer data are garbage.
  54. *
  55. * @return LSUP_Buffer pointer. It must be freed with #LSUP_buffer_free. NULL
  56. * on error.
  57. */
  58. inline LSUP_Buffer *
  59. LSUP_buffer_new (const size_t size, const void *data)
  60. {
  61. LSUP_Buffer *buf;
  62. CALLOC_GUARD (buf, NULL);
  63. if (LSUP_buffer_init (buf, size, data) != LSUP_OK) {
  64. free (buf->addr);
  65. free (buf);
  66. return NULL;
  67. }
  68. return buf;
  69. }
  70. /** @brief Dummy buffer to be used with #LSUP_buffer_init.
  71. */
  72. #define BUF_DUMMY LSUP_buffer_new (0, NULL)
  73. /** @brief Free the content of a buffer.
  74. */
  75. void LSUP_buffer_done (LSUP_Buffer *buf);
  76. /** @brief Free a buffer.
  77. */
  78. void LSUP_buffer_free (LSUP_Buffer *buf);
  79. /** @brief Hash a buffer.
  80. */
  81. inline LSUP_Key
  82. LSUP_buffer_hash (const LSUP_Buffer *buf)
  83. { return (buf == NULL) ? NULL_KEY : XXH64(buf->addr, buf->size, HASH_SEED); }
  84. /** @brief Combine hash values.
  85. *
  86. * TODO Adapt to different sizes of LSUP_Key.
  87. */
  88. inline LSUP_Key LSUP_btriple_hash (
  89. const LSUP_Buffer *b1, const LSUP_Buffer *b2)
  90. { return XXH64 (b2->addr, b2->size, XXH64 (b1->addr, b1->size, HASH_SEED)); }
  91. /** @brief Print a byte string of a given length in a human-readable format.
  92. *
  93. * The string is printed in Python style: printable characters are output
  94. * literally, and non-printable ones as hex sequences.
  95. */
  96. void LSUP_buffer_print (const LSUP_Buffer *buf);
  97. /** @brief Format a buffer into anb ASCII string.
  98. *
  99. * The string has non-printable characters escaped as "\xNN".
  100. *
  101. * @param buf[in] Buffer to convert.
  102. *
  103. * @return Formatted string. It must be freed with free().
  104. */
  105. char *
  106. LSUP_buffer_as_str (const LSUP_Buffer *buf);
  107. /** @brief Compare two buffers.
  108. *
  109. * The return value is the same as memcmp.
  110. */
  111. inline int LSUP_buffer_cmp (const LSUP_Buffer *buf1, const LSUP_Buffer *buf2)
  112. {
  113. return memcmp (
  114. buf1->addr, buf2->addr,
  115. (buf1->size > buf2->size ? buf1->size : buf2->size));
  116. }
  117. /** @brief Return whether two buffers are equal.
  118. *
  119. * This may be faster than #LSUP_buffer_cmp() because it does a size comparison
  120. * first.
  121. */
  122. inline bool LSUP_buffer_eq (const LSUP_Buffer *buf1, const LSUP_Buffer *buf2)
  123. {
  124. if (buf1->size != buf2->size) return false;
  125. return (LSUP_buffer_cmp (buf1, buf2) == 0) ? true : false;
  126. }
  127. #endif