String.cpp 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. // Copyright (C) 2009-2017, 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. Error CString::toNumber(F64& out) const
  13. {
  14. checkInit();
  15. errno = 0;
  16. out = std::strtod(m_ptr, nullptr);
  17. if(errno)
  18. {
  19. errno = 0;
  20. ANKI_UTIL_LOGE("Conversion failed");
  21. return ErrorCode::USER_DATA;
  22. }
  23. return ErrorCode::NONE;
  24. }
  25. Error CString::toNumber(F32& out) const
  26. {
  27. F64 d;
  28. ANKI_CHECK(toNumber(d));
  29. out = d;
  30. return ErrorCode::NONE;
  31. }
  32. Error CString::toNumber(I64& out) const
  33. {
  34. checkInit();
  35. errno = 0;
  36. static_assert(sizeof(long long) == sizeof(I64), "See file");
  37. out = std::strtoll(m_ptr, nullptr, 10);
  38. if(errno)
  39. {
  40. errno = 0;
  41. ANKI_UTIL_LOGE("Conversion failed");
  42. return ErrorCode::USER_DATA;
  43. }
  44. return ErrorCode::NONE;
  45. }
  46. Error CString::toNumber(I32& out) const
  47. {
  48. checkInit();
  49. errno = 0;
  50. long long i = std::strtoll(m_ptr, nullptr, 10);
  51. if(errno || i < MIN_I32 || i > MAX_I32)
  52. {
  53. errno = 0;
  54. ANKI_UTIL_LOGE("Conversion failed");
  55. return ErrorCode::USER_DATA;
  56. }
  57. out = I32(i);
  58. return ErrorCode::NONE;
  59. }
  60. Error CString::toNumber(U64& out) const
  61. {
  62. checkInit();
  63. errno = 0;
  64. static_assert(sizeof(unsigned long long) == sizeof(U64), "See file");
  65. out = std::strtoull(m_ptr, nullptr, 10);
  66. if(errno)
  67. {
  68. errno = 0;
  69. ANKI_UTIL_LOGE("Conversion failed");
  70. return ErrorCode::USER_DATA;
  71. }
  72. return ErrorCode::NONE;
  73. }
  74. Error CString::toNumber(U32& out) const
  75. {
  76. checkInit();
  77. errno = 0;
  78. unsigned long long i = std::strtoull(m_ptr, nullptr, 10);
  79. if(errno || i > MAX_U32)
  80. {
  81. errno = 0;
  82. ANKI_UTIL_LOGE("Conversion failed");
  83. return ErrorCode::USER_DATA;
  84. }
  85. out = U32(i);
  86. return ErrorCode::NONE;
  87. }
  88. String& String::operator=(StringAuto&& b)
  89. {
  90. m_data = std::move(b.m_data);
  91. return *this;
  92. }
  93. void String::create(Allocator alloc, const CStringType& cstr)
  94. {
  95. auto len = cstr.getLength();
  96. if(len > 0)
  97. {
  98. auto size = len + 1;
  99. m_data.create(alloc, size);
  100. std::memcpy(&m_data[0], &cstr[0], sizeof(Char) * size);
  101. }
  102. }
  103. void String::create(Allocator alloc, ConstIterator first, ConstIterator last)
  104. {
  105. ANKI_ASSERT(first != 0 && last != 0);
  106. auto length = last - first;
  107. m_data.create(alloc, length + 1);
  108. std::memcpy(&m_data[0], first, length);
  109. m_data[length] = '\0';
  110. }
  111. void String::create(Allocator alloc, Char c, PtrSize length)
  112. {
  113. ANKI_ASSERT(c != '\0');
  114. m_data.create(alloc, length + 1);
  115. std::memset(&m_data[0], c, length);
  116. m_data[length] = '\0';
  117. }
  118. void String::appendInternal(Allocator alloc, const Char* str, PtrSize strSize)
  119. {
  120. ANKI_ASSERT(str != nullptr);
  121. ANKI_ASSERT(strSize > 1);
  122. auto size = m_data.getSize();
  123. // Fix empty string
  124. if(size == 0)
  125. {
  126. size = 1;
  127. }
  128. DynamicArray<Char> newData;
  129. newData.create(alloc, size + strSize - 1);
  130. if(!m_data.isEmpty())
  131. {
  132. std::memcpy(&newData[0], &m_data[0], sizeof(Char) * size);
  133. }
  134. std::memcpy(&newData[size - 1], str, sizeof(Char) * strSize);
  135. m_data.destroy(alloc);
  136. m_data = std::move(newData);
  137. }
  138. void String::sprintf(Allocator alloc, CString fmt, ...)
  139. {
  140. Array<Char, 512> buffer;
  141. va_list args;
  142. va_start(args, fmt);
  143. I len = std::vsnprintf(&buffer[0], sizeof(buffer), &fmt[0], args);
  144. va_end(args);
  145. if(len < 0)
  146. {
  147. ANKI_UTIL_LOGF("vsnprintf() failed");
  148. }
  149. else if(static_cast<PtrSize>(len) >= sizeof(buffer))
  150. {
  151. I size = len + 1;
  152. m_data.create(alloc, size);
  153. va_start(args, fmt);
  154. len = std::vsnprintf(&m_data[0], size, &fmt[0], args);
  155. va_end(args);
  156. (void)len;
  157. ANKI_ASSERT((len + 1) == size);
  158. }
  159. else
  160. {
  161. // buffer was enough
  162. create(alloc, CString(&buffer[0]));
  163. }
  164. }
  165. } // end namespace anki