MelderString.h 3.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. #ifndef _melder_string_h_
  2. #define _melder_string_h_
  3. /* MelderString.h
  4. *
  5. * Copyright (C) 1992-2018 Paul Boersma
  6. *
  7. * This code is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or (at
  10. * your option) any later version.
  11. *
  12. * This code is distributed in the hope that it will be useful, but
  13. * WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  15. * See the GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this work. If not, see <http://www.gnu.org/licenses/>.
  19. */
  20. /********** STRINGS **********/
  21. /* These are functions for never having to check string boundaries again. */
  22. typedef struct {
  23. int64 length;
  24. int64 bufferSize;
  25. char16 *string; // a growing buffer, rarely shrunk (can only be freed by MelderString16_free)
  26. } MelderString16;
  27. typedef struct {
  28. int64 length;
  29. int64 bufferSize;
  30. char32 *string; // a growing buffer, rarely shrunk (can only be freed by MelderString32_free)
  31. } MelderString;
  32. void MelderString16_free (MelderString16 *me); // frees the buffer (and sets other attributes to zero)
  33. void MelderString_free (MelderString *me); // frees the buffer (and sets other attributes to zero)
  34. void MelderString16_empty (MelderString16 *me); // sets to empty string (buffer shrunk if very large)
  35. void MelderString_empty (MelderString *me); // sets to empty string (buffer shrunk if very large)
  36. void MelderString_expand (MelderString *me, int64 sizeNeeded); // increases the buffer size; there's normally no need to call this
  37. void MelderString_ncopy (MelderString *me, conststring32 source, int64 n);
  38. inline static void _recursiveTemplate_MelderString_append (MelderString *me, const MelderArg& arg) {
  39. if (arg._arg) {
  40. const char32 *newEndOfStringLocation = stp32cpy (& my string [my length], arg._arg);
  41. my length = newEndOfStringLocation - & my string [0];
  42. }
  43. }
  44. template <typename... Args>
  45. void _recursiveTemplate_MelderString_append (MelderString *me, const MelderArg& first, Args... rest) {
  46. _recursiveTemplate_MelderString_append (me, first);
  47. _recursiveTemplate_MelderString_append (me, rest...);
  48. }
  49. template <typename... Args>
  50. void MelderString_append (MelderString *me, const MelderArg& first, Args... rest) {
  51. integer extraLength = MelderArg__length (first, rest...);
  52. integer sizeNeeded = my length + extraLength + 1;
  53. if (sizeNeeded > my bufferSize)
  54. MelderString_expand (me, sizeNeeded);
  55. _recursiveTemplate_MelderString_append (me, first, rest...);
  56. }
  57. template <typename... Args>
  58. void MelderString_copy (MelderString *me, const MelderArg& first, Args... rest) {
  59. constexpr int64 FREE_THRESHOLD_BYTES = 10'000;
  60. if (my bufferSize * (int64) sizeof (char32) >= FREE_THRESHOLD_BYTES) MelderString_free (me);
  61. integer length = MelderArg__length (first, rest...);
  62. integer sizeNeeded = length + 1;
  63. if (sizeNeeded > my bufferSize)
  64. MelderString_expand (me, sizeNeeded);
  65. my length = 0;
  66. _recursiveTemplate_MelderString_append (me, first, rest...);
  67. }
  68. void MelderString16_appendCharacter (MelderString16 *me, char32 character);
  69. void MelderString_appendCharacter (MelderString *me, char32 character);
  70. void MelderString_get (MelderString *me, char32 *destination); // performs no boundary checking
  71. int64 MelderString_allocationCount ();
  72. int64 MelderString_deallocationCount ();
  73. int64 MelderString_allocationSize ();
  74. int64 MelderString_deallocationSize ();
  75. struct autoMelderString : MelderString {
  76. autoMelderString () { length = 0; bufferSize = 0; string = nullptr; }
  77. ~autoMelderString () { MelderString_free (this); }
  78. };
  79. /* End of file MelderString.h */
  80. #endif