| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329 |
- // Copyright (C) 2009-2022, Panagiotis Christopoulos Charitos and contributors.
- // All rights reserved.
- // Code licensed under the BSD License.
- // http://www.anki3d.org/LICENSE
- #include <AnKi/Util/String.h>
- #include <AnKi/Util/F16.h>
- #include <cmath> // For HUGE_VAL
- #include <climits> // For LLONG_MAX
- #include <cstdarg> // For var args
- #include <cstdlib> // For stdtod and strtol
- namespace anki {
- Error CString::toNumber(F64& out) const
- {
- checkInit();
- errno = 0;
- char* endPtr;
- out = std::strtod(m_ptr, &endPtr);
- if(errno || endPtr != m_ptr + getLength())
- {
- errno = 0;
- ANKI_UTIL_LOGE("Conversion failed: %s", m_ptr);
- return Error::USER_DATA;
- }
- return Error::NONE;
- }
- Error CString::toNumber(F32& out) const
- {
- F64 d;
- ANKI_CHECK(toNumber(d));
- out = F32(d);
- return Error::NONE;
- }
- Error CString::toNumber(F16& out) const
- {
- F64 d;
- ANKI_CHECK(toNumber(d));
- out = F16(d);
- return Error::NONE;
- }
- Error CString::toNumber(I64& out) const
- {
- checkInit();
- errno = 0;
- char* endPtr;
- static_assert(sizeof(long long) == sizeof(I64), "See file");
- out = std::strtoll(m_ptr, &endPtr, 10);
- if(errno || endPtr != m_ptr + getLength())
- {
- errno = 0;
- ANKI_UTIL_LOGE("Conversion failed: %s", m_ptr);
- return Error::USER_DATA;
- }
- return Error::NONE;
- }
- Error CString::toNumber(I8& out) const
- {
- I64 i64 = 0;
- ANKI_CHECK(toNumber(i64));
- if(i64 < MIN_I8 || i64 > MAX_I8)
- {
- ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
- return Error::USER_DATA;
- }
- out = I8(i64);
- return Error::NONE;
- }
- Error CString::toNumber(I32& out) const
- {
- I64 i64 = 0;
- ANKI_CHECK(toNumber(i64));
- if(i64 < MIN_I32 || i64 > MAX_I32)
- {
- ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
- return Error::USER_DATA;
- }
- out = I32(i64);
- return Error::NONE;
- }
- Error CString::toNumber(U64& out) const
- {
- checkInit();
- errno = 0;
- char* endPtr;
- static_assert(sizeof(unsigned long long) == sizeof(U64), "See file");
- out = std::strtoull(m_ptr, &endPtr, 10);
- if(errno || endPtr != m_ptr + getLength())
- {
- errno = 0;
- ANKI_UTIL_LOGE("Conversion failed: %s", m_ptr);
- return Error::USER_DATA;
- }
- return Error::NONE;
- }
- Error CString::toNumber(U32& out) const
- {
- U64 u64;
- ANKI_CHECK(toNumber(u64));
- if(u64 > MAX_U32)
- {
- ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
- return Error::USER_DATA;
- }
- out = U32(u64);
- return Error::NONE;
- }
- Error CString::toNumber(U8& out) const
- {
- U64 u64 = 0;
- ANKI_CHECK(toNumber(u64));
- if(u64 > MAX_U8)
- {
- ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
- return Error::USER_DATA;
- }
- out = U8(u64);
- return Error::NONE;
- }
- Error CString::toNumber(I16& out) const
- {
- I64 i64 = 0;
- ANKI_CHECK(toNumber(i64));
- if(i64 < MIN_I16 || i64 > MAX_I16)
- {
- ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
- return Error::USER_DATA;
- }
- out = I16(i64);
- return Error::NONE;
- }
- Error CString::toNumber(U16& out) const
- {
- U64 u64;
- ANKI_CHECK(toNumber(u64));
- if(u64 > MAX_U16)
- {
- ANKI_UTIL_LOGE("Conversion failed. Our of range: %s", m_ptr);
- return Error::USER_DATA;
- }
- out = U16(u64);
- return Error::NONE;
- }
- Error CString::toNumber(Bool& out) const
- {
- I32 i;
- ANKI_CHECK(toNumber(i));
- out = i != 0;
- return Error::NONE;
- }
- String& String::operator=(StringAuto&& b)
- {
- m_data = std::move(b.m_data);
- return *this;
- }
- void String::create(Allocator alloc, const CStringType& cstr)
- {
- auto len = cstr.getLength();
- auto size = len + 1;
- m_data.create(alloc, size);
- memcpy(&m_data[0], &cstr[0], sizeof(Char) * size);
- }
- void String::create(Allocator alloc, ConstIterator first, ConstIterator last)
- {
- ANKI_ASSERT(first != 0 && last != 0);
- auto length = last - first;
- m_data.create(alloc, length + 1);
- memcpy(&m_data[0], first, length);
- m_data[length] = '\0';
- }
- void String::create(Allocator alloc, Char c, PtrSize length)
- {
- ANKI_ASSERT(c != '\0');
- m_data.create(alloc, length + 1);
- memset(&m_data[0], c, length);
- m_data[length] = '\0';
- }
- void String::appendInternal(Allocator& alloc, const Char* str, PtrSize strLen)
- {
- ANKI_ASSERT(str != nullptr);
- ANKI_ASSERT(strLen > 0);
- auto size = m_data.getSize();
- // Fix empty string
- if(size == 0)
- {
- size = 1;
- }
- DynamicArray<Char, PtrSize> newData;
- newData.create(alloc, size + strLen);
- if(!m_data.isEmpty())
- {
- memcpy(&newData[0], &m_data[0], sizeof(Char) * size);
- }
- memcpy(&newData[size - 1], str, sizeof(Char) * strLen);
- newData[newData.getSize() - 1] = '\0';
- m_data.destroy(alloc);
- m_data = std::move(newData);
- }
- void String::sprintf(Allocator& alloc, const Char* fmt, va_list& args)
- {
- Array<Char, 512> buffer;
- va_list args2;
- va_copy(args2, args); // vsnprintf will alter "args". Copy it case we need to call vsnprintf in the else bellow
- I len = std::vsnprintf(&buffer[0], sizeof(buffer), fmt, args);
- if(len < 0)
- {
- ANKI_UTIL_LOGF("vsnprintf() failed");
- }
- else if(static_cast<PtrSize>(len) >= sizeof(buffer))
- {
- I size = len + 1;
- m_data.create(alloc, size);
- len = std::vsnprintf(&m_data[0], size, fmt, args2);
- ANKI_ASSERT((len + 1) == size);
- }
- else
- {
- // buffer was enough
- create(alloc, CString(&buffer[0]));
- }
- va_end(args2);
- }
- String& String::sprintf(Allocator alloc, const Char* fmt, ...)
- {
- ANKI_ASSERT(fmt);
- va_list args;
- va_start(args, fmt);
- sprintf(alloc, fmt, args);
- va_end(args);
- return *this;
- }
- String& String::replaceAll(Allocator alloc, CString from, CString to)
- {
- String tmp = {alloc, toCString()};
- const PtrSize fromLen = from.getLength();
- const PtrSize toLen = to.getLength();
- PtrSize pos = NPOS;
- while((pos = tmp.find(from)) != NPOS)
- {
- String tmp2;
- if(pos > 0)
- {
- tmp2.create(alloc, tmp.getBegin(), tmp.getBegin() + pos);
- }
- if(toLen > 0)
- {
- tmp2.append(alloc, to.getBegin(), to.getBegin() + toLen);
- }
- if(pos + fromLen < tmp.getLength())
- {
- tmp2.append(alloc, tmp.getBegin() + pos + fromLen, tmp.getEnd());
- }
- tmp.destroy(alloc);
- tmp = std::move(tmp2);
- }
- destroy(alloc);
- *this = std::move(tmp);
- return *this;
- }
- StringAuto& StringAuto::sprintf(const Char* fmt, ...)
- {
- va_list args;
- va_start(args, fmt);
- Base::sprintf(m_alloc, fmt, args);
- va_end(args);
- return *this;
- }
- } // end namespace anki
|