String.cpp 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. // Copyright (C) 2009-2021, 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 <AnKi/Util/F16.h>
  7. #include <cmath> // For HUGE_VAL
  8. #include <climits> // For LLONG_MAX
  9. #include <cstdarg> // For var args
  10. #include <cstdlib> // For stdtod and strtol
  11. namespace anki
  12. {
  13. Error CString::toNumber(F64& out) const
  14. {
  15. checkInit();
  16. errno = 0;
  17. char* endPtr;
  18. out = std::strtod(m_ptr, &endPtr);
  19. if(errno || endPtr != m_ptr + getLength())
  20. {
  21. errno = 0;
  22. ANKI_UTIL_LOGE("Conversion failed: %s", m_ptr);
  23. return Error::USER_DATA;
  24. }
  25. return Error::NONE;
  26. }
  27. Error CString::toNumber(F32& out) const
  28. {
  29. F64 d;
  30. ANKI_CHECK(toNumber(d));
  31. out = F32(d);
  32. return Error::NONE;
  33. }
  34. Error CString::toNumber(F16& out) const
  35. {
  36. F64 d;
  37. ANKI_CHECK(toNumber(d));
  38. out = F16(d);
  39. return Error::NONE;
  40. }
  41. Error CString::toNumber(I64& out) const
  42. {
  43. checkInit();
  44. errno = 0;
  45. char* endPtr;
  46. static_assert(sizeof(long long) == sizeof(I64), "See file");
  47. out = std::strtoll(m_ptr, &endPtr, 10);
  48. if(errno || endPtr != m_ptr + getLength())
  49. {
  50. errno = 0;
  51. ANKI_UTIL_LOGE("Conversion failed: %s", m_ptr);
  52. return Error::USER_DATA;
  53. }
  54. return Error::NONE;
  55. }
  56. Error CString::toNumber(I8& out) const
  57. {
  58. I64 i64 = 0;
  59. ANKI_CHECK(toNumber(i64));
  60. if(i64 < MIN_I8 || i64 > MAX_I8)
  61. {
  62. ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
  63. return Error::USER_DATA;
  64. }
  65. out = I8(i64);
  66. return Error::NONE;
  67. }
  68. Error CString::toNumber(I32& out) const
  69. {
  70. I64 i64 = 0;
  71. ANKI_CHECK(toNumber(i64));
  72. if(i64 < MIN_I32 || i64 > MAX_I32)
  73. {
  74. ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
  75. return Error::USER_DATA;
  76. }
  77. out = I32(i64);
  78. return Error::NONE;
  79. }
  80. Error CString::toNumber(U64& out) const
  81. {
  82. checkInit();
  83. errno = 0;
  84. char* endPtr;
  85. static_assert(sizeof(unsigned long long) == sizeof(U64), "See file");
  86. out = std::strtoull(m_ptr, &endPtr, 10);
  87. if(errno || endPtr != m_ptr + getLength())
  88. {
  89. errno = 0;
  90. ANKI_UTIL_LOGE("Conversion failed: %s", m_ptr);
  91. return Error::USER_DATA;
  92. }
  93. return Error::NONE;
  94. }
  95. Error CString::toNumber(U32& out) const
  96. {
  97. U64 u64;
  98. ANKI_CHECK(toNumber(u64));
  99. if(u64 > MAX_U32)
  100. {
  101. ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
  102. return Error::USER_DATA;
  103. }
  104. out = U32(u64);
  105. return Error::NONE;
  106. }
  107. Error CString::toNumber(U8& out) const
  108. {
  109. U64 u64 = 0;
  110. ANKI_CHECK(toNumber(u64));
  111. if(u64 > MAX_U8)
  112. {
  113. ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
  114. return Error::USER_DATA;
  115. }
  116. out = U8(u64);
  117. return Error::NONE;
  118. }
  119. Error CString::toNumber(I16& out) const
  120. {
  121. I64 i64 = 0;
  122. ANKI_CHECK(toNumber(i64));
  123. if(i64 < MIN_I16 || i64 > MAX_I16)
  124. {
  125. ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
  126. return Error::USER_DATA;
  127. }
  128. out = I16(i64);
  129. return Error::NONE;
  130. }
  131. Error CString::toNumber(U16& out) const
  132. {
  133. U64 u64;
  134. ANKI_CHECK(toNumber(u64));
  135. if(u64 > MAX_U16)
  136. {
  137. ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
  138. return Error::USER_DATA;
  139. }
  140. out = U16(u64);
  141. return Error::NONE;
  142. }
  143. Error CString::toNumber(Bool& out) const
  144. {
  145. I32 i;
  146. ANKI_CHECK(toNumber(i));
  147. out = i != 0;
  148. return Error::NONE;
  149. }
  150. String& String::operator=(StringAuto&& b)
  151. {
  152. m_data = std::move(b.m_data);
  153. return *this;
  154. }
  155. void String::create(Allocator alloc, const CStringType& cstr)
  156. {
  157. auto len = cstr.getLength();
  158. auto size = len + 1;
  159. m_data.create(alloc, size);
  160. memcpy(&m_data[0], &cstr[0], sizeof(Char) * size);
  161. }
  162. void String::create(Allocator alloc, ConstIterator first, ConstIterator last)
  163. {
  164. ANKI_ASSERT(first != 0 && last != 0);
  165. auto length = last - first;
  166. m_data.create(alloc, length + 1);
  167. memcpy(&m_data[0], first, length);
  168. m_data[length] = '\0';
  169. }
  170. void String::create(Allocator alloc, Char c, PtrSize length)
  171. {
  172. ANKI_ASSERT(c != '\0');
  173. m_data.create(alloc, length + 1);
  174. memset(&m_data[0], c, length);
  175. m_data[length] = '\0';
  176. }
  177. void String::appendInternal(Allocator& alloc, const Char* str, PtrSize strLen)
  178. {
  179. ANKI_ASSERT(str != nullptr);
  180. ANKI_ASSERT(strLen > 0);
  181. auto size = m_data.getSize();
  182. // Fix empty string
  183. if(size == 0)
  184. {
  185. size = 1;
  186. }
  187. DynamicArray<Char, PtrSize> newData;
  188. newData.create(alloc, size + strLen);
  189. if(!m_data.isEmpty())
  190. {
  191. memcpy(&newData[0], &m_data[0], sizeof(Char) * size);
  192. }
  193. memcpy(&newData[size - 1], str, sizeof(Char) * strLen);
  194. newData[newData.getSize() - 1] = '\0';
  195. m_data.destroy(alloc);
  196. m_data = std::move(newData);
  197. }
  198. String& String::sprintf(Allocator alloc, CString fmt, ...)
  199. {
  200. Array<Char, 512> buffer;
  201. va_list args;
  202. va_start(args, fmt);
  203. I len = std::vsnprintf(&buffer[0], sizeof(buffer), &fmt[0], args);
  204. va_end(args);
  205. if(len < 0)
  206. {
  207. ANKI_UTIL_LOGF("vsnprintf() failed");
  208. }
  209. else if(static_cast<PtrSize>(len) >= sizeof(buffer))
  210. {
  211. I size = len + 1;
  212. m_data.create(alloc, size);
  213. va_start(args, fmt);
  214. len = std::vsnprintf(&m_data[0], size, &fmt[0], args);
  215. va_end(args);
  216. (void)len;
  217. ANKI_ASSERT((len + 1) == size);
  218. }
  219. else
  220. {
  221. // buffer was enough
  222. create(alloc, CString(&buffer[0]));
  223. }
  224. return *this;
  225. }
  226. String& String::replaceAll(Allocator alloc, CString from, CString to)
  227. {
  228. String tmp = {alloc, toCString()};
  229. const PtrSize fromLen = from.getLength();
  230. const PtrSize toLen = to.getLength();
  231. PtrSize pos = NPOS;
  232. while((pos = tmp.find(from)) != NPOS)
  233. {
  234. String tmp2;
  235. if(pos > 0)
  236. {
  237. tmp2.create(alloc, tmp.getBegin(), tmp.getBegin() + pos);
  238. }
  239. if(toLen > 0)
  240. {
  241. tmp2.append(alloc, to.getBegin(), to.getBegin() + toLen);
  242. }
  243. if(pos + fromLen < tmp.getLength())
  244. {
  245. tmp2.append(alloc, tmp.getBegin() + pos + fromLen, tmp.getEnd());
  246. }
  247. tmp.destroy(alloc);
  248. tmp = std::move(tmp2);
  249. }
  250. destroy(alloc);
  251. *this = std::move(tmp);
  252. return *this;
  253. }
  254. } // end namespace anki