String.cpp 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. // Copyright (C) 2009-2016, Panagiotis Christopoulos Charitos and contributors.
  2. // All rights reserved.
  3. // Code licensed under the BSD License.
  4. // http://www.anki3d.org/LICENSE
  5. #include <anki/util/String.h>
  6. #include <cmath> // For HUGE_VAL
  7. #include <climits> // For LLONG_MAX
  8. #include <cstdarg> // For var args
  9. #include <cstdlib> // For stdtod and strtol
  10. namespace anki
  11. {
  12. //==============================================================================
  13. // CString =
  14. //==============================================================================
  15. //==============================================================================
  16. Error CString::toF64(F64& out) const
  17. {
  18. checkInit();
  19. Error err = ErrorCode::NONE;
  20. out = std::strtod(m_ptr, nullptr);
  21. if(out == HUGE_VAL)
  22. {
  23. ANKI_LOGE("Conversion failed");
  24. err = ErrorCode::USER_DATA;
  25. }
  26. return err;
  27. }
  28. //==============================================================================
  29. Error CString::toF32(F32& out) const
  30. {
  31. F64 d;
  32. Error err = toF64(d);
  33. if(!err)
  34. {
  35. out = d;
  36. }
  37. return err;
  38. }
  39. //==============================================================================
  40. Error CString::toI64(I64& out) const
  41. {
  42. checkInit();
  43. Error err = ErrorCode::NONE;
  44. out = std::strtoll(m_ptr, nullptr, 10);
  45. if(out == LLONG_MAX || out == LLONG_MIN)
  46. {
  47. ANKI_LOGE("Conversion failed");
  48. err = ErrorCode::USER_DATA;
  49. }
  50. return err;
  51. }
  52. //==============================================================================
  53. // String =
  54. //==============================================================================
  55. //==============================================================================
  56. void String::create(Allocator alloc, const CStringType& cstr)
  57. {
  58. auto len = cstr.getLength();
  59. if(len > 0)
  60. {
  61. auto size = len + 1;
  62. m_data.create(alloc, size);
  63. std::memcpy(&m_data[0], &cstr[0], sizeof(Char) * size);
  64. }
  65. }
  66. //==============================================================================
  67. void String::create(Allocator alloc, ConstIterator first, ConstIterator last)
  68. {
  69. ANKI_ASSERT(first != 0 && last != 0);
  70. auto length = last - first;
  71. m_data.create(alloc, length + 1);
  72. std::memcpy(&m_data[0], first, length);
  73. m_data[length] = '\0';
  74. }
  75. //==============================================================================
  76. void String::create(Allocator alloc, Char c, PtrSize length)
  77. {
  78. ANKI_ASSERT(c != '\0');
  79. m_data.create(alloc, length + 1);
  80. std::memset(&m_data[0], c, length);
  81. m_data[length] = '\0';
  82. }
  83. //==============================================================================
  84. void String::appendInternal(Allocator alloc, const Char* str, PtrSize strSize)
  85. {
  86. ANKI_ASSERT(str != nullptr);
  87. ANKI_ASSERT(strSize > 1);
  88. auto size = m_data.getSize();
  89. // Fix empty string
  90. if(size == 0)
  91. {
  92. size = 1;
  93. }
  94. DynamicArray<Char> newData;
  95. newData.create(alloc, size + strSize - 1);
  96. if(!m_data.isEmpty())
  97. {
  98. std::memcpy(&newData[0], &m_data[0], sizeof(Char) * size);
  99. }
  100. std::memcpy(&newData[size - 1], str, sizeof(Char) * strSize);
  101. m_data.destroy(alloc);
  102. m_data = std::move(newData);
  103. }
  104. //==============================================================================
  105. void String::sprintf(Allocator alloc, CString fmt, ...)
  106. {
  107. Array<Char, 512> buffer;
  108. va_list args;
  109. va_start(args, fmt);
  110. I len = std::vsnprintf(&buffer[0], sizeof(buffer), &fmt[0], args);
  111. va_end(args);
  112. if(len < 0)
  113. {
  114. ANKI_LOGF("vsnprintf() failed");
  115. }
  116. else if(static_cast<PtrSize>(len) >= sizeof(buffer))
  117. {
  118. I size = len + 1;
  119. m_data.create(alloc, size);
  120. va_start(args, fmt);
  121. len = std::vsnprintf(&m_data[0], size, &fmt[0], args);
  122. va_end(args);
  123. (void)len;
  124. ANKI_ASSERT((len + 1) == size);
  125. }
  126. else
  127. {
  128. // buffer was enough
  129. create(alloc, CString(&buffer[0]));
  130. }
  131. }
  132. } // end namespace anki