2020-10-27 23:22:25 +01:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
2021-12-26 18:47:58 +01:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
2020-10-27 23:22:25 +01:00
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2021-12-26 18:47:58 +01:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2020-10-27 23:22:25 +01:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2023-01-05 14:21:23 +01:00
|
|
|
#include "common/str-base.h"
|
2020-10-27 23:22:25 +01:00
|
|
|
#include "common/hash-str.h"
|
|
|
|
#include "common/list.h"
|
|
|
|
#include "common/memorypool.h"
|
2021-06-29 15:12:20 +02:00
|
|
|
#include "common/textconsole.h"
|
2020-10-27 23:22:25 +01:00
|
|
|
#include "common/util.h"
|
|
|
|
#include "common/mutex.h"
|
|
|
|
|
|
|
|
namespace Common {
|
|
|
|
|
|
|
|
#define TEMPLATE template<class T>
|
|
|
|
#define BASESTRING BaseString<T>
|
|
|
|
|
|
|
|
MemoryPool *g_refCountPool = nullptr; // FIXME: This is never freed right now
|
|
|
|
#ifndef SCUMMVM_UTIL
|
|
|
|
Mutex *g_refCountPoolMutex = nullptr;
|
|
|
|
|
|
|
|
void lockMemoryPoolMutex() {
|
|
|
|
// The Mutex class can only be used once g_system is set and initialized,
|
|
|
|
// but we may use the String class earlier than that (it is for example
|
|
|
|
// used in the OSystem_POSIX constructor). However in those early stages
|
|
|
|
// we can hope we don't have multiple threads either.
|
|
|
|
if (!g_system || !g_system->backendInitialized())
|
|
|
|
return;
|
|
|
|
if (!g_refCountPoolMutex)
|
|
|
|
g_refCountPoolMutex = new Mutex();
|
|
|
|
g_refCountPoolMutex->lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void unlockMemoryPoolMutex() {
|
|
|
|
if (g_refCountPoolMutex)
|
|
|
|
g_refCountPoolMutex->unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::releaseMemoryPoolMutex() {
|
|
|
|
if (g_refCountPoolMutex){
|
|
|
|
delete g_refCountPoolMutex;
|
|
|
|
g_refCountPoolMutex = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static uint32 computeCapacity(uint32 len) {
|
|
|
|
// By default, for the capacity we use the next multiple of 32
|
|
|
|
return ((len + 32 - 1) & ~0x1F);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE
|
|
|
|
BASESTRING::BaseString(const BASESTRING &str)
|
2021-04-15 21:20:04 +02:00
|
|
|
: _size(str._size) {
|
2020-10-27 23:22:25 +01:00
|
|
|
if (str.isStorageIntern()) {
|
|
|
|
// String in internal storage: just copy it
|
|
|
|
memcpy(_storage, str._storage, _builtinCapacity * sizeof(value_type));
|
|
|
|
_str = _storage;
|
|
|
|
} else {
|
|
|
|
// String in external storage: use refcount mechanism
|
|
|
|
str.incRefCount();
|
|
|
|
_extern._refCount = str._extern._refCount;
|
|
|
|
_extern._capacity = str._extern._capacity;
|
|
|
|
_str = str._str;
|
|
|
|
}
|
|
|
|
assert(_str != nullptr);
|
|
|
|
}
|
|
|
|
|
2022-09-24 17:49:58 -04:00
|
|
|
|
|
|
|
TEMPLATE
|
|
|
|
BASESTRING::BaseString(BASESTRING &&str) : _size(str._size) {
|
|
|
|
if (str.isStorageIntern()) {
|
|
|
|
// String in internal storage: copy it
|
|
|
|
memcpy(_storage, str._storage, _builtinCapacity * sizeof(value_type));
|
|
|
|
_str = _storage;
|
|
|
|
} else {
|
|
|
|
// String in external storage: Take the reference
|
|
|
|
_extern = str._extern;
|
|
|
|
_str = str._str;
|
|
|
|
}
|
|
|
|
|
|
|
|
str._str = str._storage;
|
|
|
|
str._storage[0] = 0;
|
|
|
|
str._size = 0;
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE BASESTRING::BaseString(const value_type *str) : _size(0), _str(_storage) {
|
|
|
|
if (str == nullptr) {
|
|
|
|
_storage[0] = 0;
|
|
|
|
_size = 0;
|
2023-08-13 10:14:13 -04:00
|
|
|
} else
|
|
|
|
initWithValueTypeStr(str, cStrLen(str));
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE BASESTRING::BaseString(const value_type *str, uint32 len) : _size(0), _str(_storage) {
|
|
|
|
initWithValueTypeStr(str, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE BASESTRING::BaseString(const value_type *beginP, const value_type *endP) : _size(0), _str(_storage) {
|
|
|
|
assert(endP >= beginP);
|
|
|
|
initWithValueTypeStr(beginP, endP - beginP);
|
|
|
|
}
|
|
|
|
|
2020-11-05 23:39:48 +00:00
|
|
|
TEMPLATE BASESTRING::~BaseString() {
|
|
|
|
decRefCount(_extern._refCount);
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE void BASESTRING::ensureCapacity(uint32 new_size, bool keep_old) {
|
|
|
|
bool isShared;
|
|
|
|
uint32 curCapacity, newCapacity;
|
|
|
|
value_type *newStorage;
|
|
|
|
int *oldRefCount = _extern._refCount;
|
|
|
|
|
|
|
|
if (isStorageIntern()) {
|
|
|
|
isShared = false;
|
|
|
|
curCapacity = _builtinCapacity;
|
|
|
|
} else {
|
|
|
|
isShared = (oldRefCount && *oldRefCount > 1);
|
|
|
|
curCapacity = _extern._capacity;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Special case: If there is enough space, and we do not share
|
|
|
|
// the storage, then there is nothing to do.
|
|
|
|
if (!isShared && new_size < curCapacity)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (isShared && new_size < _builtinCapacity) {
|
|
|
|
// We share the storage, but there is enough internal storage: Use that.
|
|
|
|
newStorage = _storage;
|
|
|
|
newCapacity = _builtinCapacity;
|
|
|
|
} else {
|
|
|
|
// We need to allocate storage on the heap!
|
|
|
|
|
|
|
|
// Compute a suitable new capacity limit
|
|
|
|
// If the current capacity is sufficient we use the same capacity
|
|
|
|
if (new_size < curCapacity)
|
|
|
|
newCapacity = curCapacity;
|
|
|
|
else
|
|
|
|
newCapacity = MAX(curCapacity * 2, computeCapacity(new_size + 1));
|
|
|
|
|
|
|
|
// Allocate new storage
|
|
|
|
newStorage = new value_type[newCapacity];
|
|
|
|
assert(newStorage);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy old data if needed, elsewise reset the new storage.
|
|
|
|
if (keep_old) {
|
|
|
|
assert(_size < newCapacity);
|
|
|
|
memcpy(newStorage, _str, (_size + 1) * sizeof(value_type));
|
|
|
|
} else {
|
|
|
|
_size = 0;
|
|
|
|
newStorage[0] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Release hold on the old storage ...
|
|
|
|
decRefCount(oldRefCount);
|
|
|
|
|
|
|
|
// ... in favor of the new storage
|
|
|
|
_str = newStorage;
|
|
|
|
|
|
|
|
if (!isStorageIntern()) {
|
|
|
|
// Set the ref count & capacity if we use an external storage.
|
|
|
|
// It is important to do this *after* copying any old content,
|
|
|
|
// else we would override data that has not yet been copied!
|
|
|
|
_extern._refCount = nullptr;
|
|
|
|
_extern._capacity = newCapacity;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE
|
|
|
|
void BASESTRING::incRefCount() const {
|
|
|
|
assert(!isStorageIntern());
|
|
|
|
if (_extern._refCount == nullptr) {
|
2021-03-31 23:28:11 +01:00
|
|
|
#ifndef SCUMMVM_UTIL
|
2021-03-31 22:50:11 +01:00
|
|
|
lockMemoryPoolMutex();
|
2021-03-31 23:28:11 +01:00
|
|
|
#endif
|
2020-10-27 23:22:25 +01:00
|
|
|
if (g_refCountPool == nullptr) {
|
|
|
|
g_refCountPool = new MemoryPool(sizeof(int));
|
|
|
|
assert(g_refCountPool);
|
|
|
|
}
|
|
|
|
|
|
|
|
_extern._refCount = (int *)g_refCountPool->allocChunk();
|
2021-03-31 23:28:11 +01:00
|
|
|
#ifndef SCUMMVM_UTIL
|
2021-03-31 22:50:11 +01:00
|
|
|
unlockMemoryPoolMutex();
|
2021-03-31 23:28:11 +01:00
|
|
|
#endif
|
2020-10-27 23:22:25 +01:00
|
|
|
*_extern._refCount = 2;
|
|
|
|
} else {
|
|
|
|
++(*_extern._refCount);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE
|
|
|
|
void BASESTRING::decRefCount(int *oldRefCount) {
|
|
|
|
if (isStorageIntern())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (oldRefCount) {
|
|
|
|
--(*oldRefCount);
|
|
|
|
}
|
|
|
|
if (!oldRefCount || *oldRefCount <= 0) {
|
|
|
|
// The ref count reached zero, so we free the string storage
|
|
|
|
// and the ref count storage.
|
|
|
|
if (oldRefCount) {
|
2021-03-31 23:28:11 +01:00
|
|
|
#ifndef SCUMMVM_UTIL
|
2021-03-31 22:50:11 +01:00
|
|
|
lockMemoryPoolMutex();
|
2021-03-31 23:28:11 +01:00
|
|
|
#endif
|
2020-10-27 23:22:25 +01:00
|
|
|
assert(g_refCountPool);
|
|
|
|
g_refCountPool->freeChunk(oldRefCount);
|
2021-03-31 23:28:11 +01:00
|
|
|
#ifndef SCUMMVM_UTIL
|
2021-03-31 22:50:11 +01:00
|
|
|
unlockMemoryPoolMutex();
|
2021-03-31 23:28:11 +01:00
|
|
|
#endif
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
// Coverity thinks that we always free memory, as it assumes
|
|
|
|
// (correctly) that there are cases when oldRefCount == 0
|
|
|
|
// Thus, DO NOT COMPILE, trick it and shut tons of false positives
|
|
|
|
#ifndef __COVERITY__
|
|
|
|
delete[] _str;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Even though _str points to a freed memory block now,
|
|
|
|
// we do not change its value, because any code that calls
|
|
|
|
// decRefCount will have to do this afterwards anyway.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::initWithValueTypeStr(const value_type *str, uint32 len) {
|
|
|
|
assert(str);
|
|
|
|
|
|
|
|
_storage[0] = 0;
|
|
|
|
|
|
|
|
_size = len;
|
|
|
|
|
|
|
|
if (len >= _builtinCapacity) {
|
|
|
|
// Not enough internal storage, so allocate more
|
|
|
|
_extern._capacity = computeCapacity(len + 1);
|
|
|
|
_extern._refCount = nullptr;
|
|
|
|
_str = new value_type[_extern._capacity];
|
|
|
|
assert(_str != nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy the string into the storage area
|
|
|
|
memmove(_str, str, len * sizeof(value_type));
|
|
|
|
_str[len] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::equals(const BaseString &x) const {
|
|
|
|
if (this == &x || _str == x._str) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (x.size() != _size) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return !memcmp(_str, x._str, _size * sizeof(value_type));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::equals(const value_type *ptr) const {
|
|
|
|
if (_str == ptr) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint i = 0;
|
|
|
|
|
|
|
|
for (; i < _size && *ptr; i++, ptr++) {
|
|
|
|
if (_str[i] != *ptr)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == _size && *ptr == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::equalsC(const char *ptr) const {
|
|
|
|
uint i = 0;
|
|
|
|
|
|
|
|
for (; i < _size && *ptr; i++, ptr++) {
|
2020-12-21 18:24:46 -05:00
|
|
|
if (_str[i] != (T)*ptr)
|
2020-10-27 23:22:25 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == _size && *ptr == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator==(const BaseString &x) const {
|
|
|
|
return equals(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator==(const value_type *x) const {
|
2020-11-06 10:44:25 +01:00
|
|
|
return equals(x);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator!=(const BaseString &x) const {
|
|
|
|
return !equals(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator!=(const value_type *x) const {
|
2020-11-06 10:44:25 +01:00
|
|
|
return !equals(x);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE int BASESTRING::compareTo(const BaseString &x) const {
|
2024-01-14 14:05:24 +01:00
|
|
|
if (_size == 0 &&
|
|
|
|
x.size() == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-01-09 09:52:55 +00:00
|
|
|
int n = cMemCmp(_str, x._str, MIN<uint32>(_size, x.size()));
|
2023-12-30 21:38:06 +00:00
|
|
|
if (n != 0)
|
|
|
|
return n;
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
if (_size < x.size())
|
|
|
|
return -1;
|
|
|
|
if (_size == x.size())
|
|
|
|
return 0;
|
|
|
|
return +1;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE int BASESTRING::compareTo(const value_type *ptr) const {
|
|
|
|
uint32 i = 0;
|
|
|
|
for (; i < _size && *ptr; ++i, ptr++) {
|
|
|
|
uint32 sc = _str[i];
|
|
|
|
uint32 xc = *ptr;
|
|
|
|
if (sc < xc)
|
|
|
|
return -1;
|
|
|
|
else if (sc > xc)
|
|
|
|
return +1;
|
|
|
|
}
|
|
|
|
if (i == _size && *ptr == 0)
|
|
|
|
return 0;
|
|
|
|
if (*ptr == 0)
|
|
|
|
return +1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE int BASESTRING::compareToC(const char *ptr) const {
|
|
|
|
uint32 i = 0;
|
|
|
|
for (; i < _size && *ptr; ++i, ptr++) {
|
|
|
|
uint32 sc = _str[i];
|
|
|
|
uint32 xc = *ptr;
|
|
|
|
if (sc < xc)
|
|
|
|
return -1;
|
|
|
|
else if (sc > xc)
|
|
|
|
return +1;
|
|
|
|
}
|
|
|
|
if (i == _size && *ptr == 0)
|
|
|
|
return 0;
|
|
|
|
if (*ptr == 0)
|
|
|
|
return +1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator<(const BaseString &x) const {
|
|
|
|
return compareTo(x) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator<=(const BaseString &x) const {
|
|
|
|
return compareTo(x) <= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator>(const BaseString &x) const {
|
|
|
|
return compareTo(x) > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator>=(const BaseString &x) const {
|
|
|
|
return compareTo(x) >= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator<(const value_type *x) const {
|
|
|
|
return compareTo(x) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator<=(const value_type *x) const {
|
|
|
|
return compareTo(x) <= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator>(const value_type *x) const {
|
|
|
|
return compareTo(x) > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::operator>=(const value_type *x) const {
|
|
|
|
return compareTo(x) >= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::contains(value_type x) const {
|
2023-12-30 21:38:06 +00:00
|
|
|
return cMemChr(_str, x, _size) != nullptr;
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::contains(const BaseString &otherString) const {
|
|
|
|
if (empty() || otherString.empty() || _size < otherString.size()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-06-20 21:11:13 +03:00
|
|
|
for (const_iterator cur = begin(); cur != end(); ++cur) {
|
|
|
|
uint i = 0;
|
|
|
|
while (true) {
|
|
|
|
if (i == otherString.size()) {
|
2020-10-27 23:22:25 +01:00
|
|
|
return true;
|
2021-06-20 21:11:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (cur[i] != otherString[i]) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
++i;
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::insertChar(value_type c, uint32 p) {
|
|
|
|
assert(p <= _size);
|
2023-12-30 16:24:47 +00:00
|
|
|
assignInsert(c, p);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::deleteChar(uint32 p) {
|
|
|
|
assert(p < _size);
|
2023-12-30 17:29:02 +00:00
|
|
|
erase(p, 1);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
2023-09-01 13:16:20 +02:00
|
|
|
TEMPLATE void BASESTRING::chop(uint32 len) {
|
2023-12-30 17:29:02 +00:00
|
|
|
if (_size > len) {
|
|
|
|
uint32 newSize = _size - len;
|
|
|
|
|
|
|
|
makeUnique();
|
|
|
|
_str[newSize] = 0;
|
|
|
|
_size = newSize;
|
|
|
|
} else if (_size > 0) {
|
|
|
|
clear();
|
|
|
|
}
|
2023-09-01 13:16:20 +02:00
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE void BASESTRING::erase(uint32 p, uint32 len) {
|
2020-11-02 21:49:05 -08:00
|
|
|
if (len == 0)
|
|
|
|
return;
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
assert(p < _size);
|
|
|
|
|
|
|
|
// If len == npos or p + len is over the end, remove all the way to the end
|
|
|
|
if (len == npos || p + len >= _size) {
|
2023-12-30 17:29:02 +00:00
|
|
|
// If p == 0, remove the entire string
|
|
|
|
if (p == 0) {
|
|
|
|
clear();
|
|
|
|
} else {
|
|
|
|
makeUnique();
|
|
|
|
// Delete char at p as well. So _size = (p - 1) + 1
|
|
|
|
_size = p;
|
|
|
|
// Null terminate
|
|
|
|
_str[_size] = 0;
|
|
|
|
}
|
2020-10-27 23:22:25 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-12-30 17:29:02 +00:00
|
|
|
makeUnique();
|
2020-10-27 23:22:25 +01:00
|
|
|
_size -= len;
|
2023-12-30 17:29:02 +00:00
|
|
|
memmove(_str + p, _str + p + len, ((_size - p) + 1) * sizeof(value_type));
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE typename BASESTRING::iterator BASESTRING::erase(iterator it) {
|
|
|
|
this->deleteChar(it - _str);
|
|
|
|
return it;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::clear() {
|
|
|
|
decRefCount(_extern._refCount);
|
|
|
|
|
|
|
|
_size = 0;
|
|
|
|
_str = _storage;
|
|
|
|
_storage[0] = 0;
|
|
|
|
}
|
|
|
|
|
2021-05-04 11:45:03 +03:00
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE void BASESTRING::setChar(value_type c, uint32 p) {
|
|
|
|
assert(p < _size);
|
|
|
|
|
|
|
|
makeUnique();
|
|
|
|
_str[p] = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::assign(const BaseString &str) {
|
|
|
|
if (&str == this)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (str.isStorageIntern()) {
|
|
|
|
decRefCount(_extern._refCount);
|
|
|
|
_size = str._size;
|
|
|
|
_str = _storage;
|
|
|
|
memcpy(_str, str._str, (_size + 1) * sizeof(value_type));
|
|
|
|
} else {
|
|
|
|
str.incRefCount();
|
|
|
|
decRefCount(_extern._refCount);
|
|
|
|
|
|
|
|
_extern._refCount = str._extern._refCount;
|
|
|
|
_extern._capacity = str._extern._capacity;
|
|
|
|
_size = str._size;
|
|
|
|
_str = str._str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-24 17:49:58 -04:00
|
|
|
TEMPLATE void BASESTRING::assign(BaseString &&str) {
|
|
|
|
if (&str == this)
|
|
|
|
return;
|
|
|
|
|
|
|
|
decRefCount(_extern._refCount);
|
|
|
|
|
|
|
|
if (str.isStorageIntern()) {
|
|
|
|
_str = _storage;
|
|
|
|
memcpy(_str, str._str, _builtinCapacity * sizeof(value_type));
|
|
|
|
} else {
|
|
|
|
_extern = str._extern;
|
|
|
|
_str = str._str;
|
|
|
|
}
|
|
|
|
|
|
|
|
_size = str._size;
|
|
|
|
|
|
|
|
str._storage[0] = 0;
|
|
|
|
str._size = 0;
|
|
|
|
str._str = str._storage;
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE void BASESTRING::assign(value_type c) {
|
|
|
|
decRefCount(_extern._refCount);
|
|
|
|
_str = _storage;
|
|
|
|
|
|
|
|
_str[0] = c;
|
|
|
|
_str[1] = 0;
|
|
|
|
|
|
|
|
_size = (c == 0) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::insertString(const value_type *s, uint32 p) {
|
2023-12-30 16:24:47 +00:00
|
|
|
assignInsert(s, p);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::insertString(const BaseString &s, uint32 p) {
|
2023-12-30 16:24:47 +00:00
|
|
|
assignInsert(s, p);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE uint32 BASESTRING::find(value_type x, uint32 pos) const {
|
2023-12-30 21:38:06 +00:00
|
|
|
const value_type *p = (pos >= _size) ? nullptr : cMemChr(_str + pos, x, _size - pos);
|
|
|
|
return p ? p - _str : npos;
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE uint32 BASESTRING::find(const BaseString &str, uint32 pos) const {
|
|
|
|
if (pos >= _size) {
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
const value_type *strP = str.c_str();
|
|
|
|
|
|
|
|
for (const_iterator cur = begin() + pos; cur != end(); ++cur) {
|
|
|
|
uint i = 0;
|
|
|
|
while (true) {
|
|
|
|
if (!strP[i]) {
|
|
|
|
return cur - begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cur[i] != strP[i]) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::find(const value_type *strP, uint32 pos) const {
|
2020-11-05 23:10:52 +00:00
|
|
|
if (pos >= _size) {
|
2020-10-27 23:22:25 +01:00
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const_iterator cur = begin() + pos; cur != end(); ++cur) {
|
|
|
|
uint i = 0;
|
|
|
|
while (true) {
|
|
|
|
if (!strP[i]) {
|
|
|
|
return cur - begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cur[i] != strP[i]) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
2023-12-30 22:03:29 +00:00
|
|
|
TEMPLATE size_t BASESTRING::rfind(const value_type *s) const {
|
|
|
|
int sLen = cStrLen(s);
|
|
|
|
|
|
|
|
for (int idx = (int)_size - sLen; idx >= 0; --idx) {
|
|
|
|
if (!memcmp(_str + idx, s, sLen * sizeof(value_type)))
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::rfind(value_type c, size_t pos) const {
|
|
|
|
if (pos == npos || pos > _size)
|
|
|
|
pos = _size;
|
|
|
|
else
|
|
|
|
++pos;
|
|
|
|
|
|
|
|
while (pos > 0) {
|
|
|
|
--pos;
|
|
|
|
if ((*this)[pos] == c)
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findFirstOf(value_type c, size_t pos) const {
|
|
|
|
const value_type *strP = (pos >= _size) ? nullptr : cMemChr(_str + pos, c, _size - pos);
|
|
|
|
return strP ? strP - _str : npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findFirstOf(const value_type *chars, size_t pos) const {
|
|
|
|
uint32 charsLen = cStrLen(chars);
|
|
|
|
|
|
|
|
for (uint idx = pos; idx < _size; ++idx) {
|
|
|
|
if (cMemChr(chars, (*this)[idx], charsLen))
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findLastOf(value_type c, size_t pos) const {
|
|
|
|
int start = (pos == npos) ? (int)_size - 1 : MIN((int)_size - 1, (int)pos);
|
|
|
|
for (int idx = start; idx >= 0; --idx) {
|
|
|
|
if ((*this)[idx] == c)
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findLastOf(const value_type *chars, size_t pos) const {
|
|
|
|
uint32 charsLen = cStrLen(chars);
|
|
|
|
|
|
|
|
int start = (pos == npos) ? (int)_size - 1 : MIN((int)_size - 1, (int)pos);
|
|
|
|
for (int idx = start; idx >= 0; --idx) {
|
|
|
|
if (cMemChr(chars, (*this)[idx], charsLen))
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findFirstNotOf(value_type c, size_t pos) const {
|
|
|
|
for (uint idx = pos; idx < _size; ++idx) {
|
|
|
|
if ((*this)[idx] != c)
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findFirstNotOf(const value_type *chars, size_t pos) const {
|
|
|
|
uint32 charsLen = cStrLen(chars);
|
|
|
|
|
|
|
|
for (uint idx = pos; idx < _size; ++idx) {
|
|
|
|
if (!cMemChr(chars, (*this)[idx], charsLen))
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findLastNotOf(value_type c) const {
|
|
|
|
for (int idx = (int)_size - 1; idx >= 0; --idx) {
|
|
|
|
if ((*this)[idx] != c)
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE size_t BASESTRING::findLastNotOf(const value_type *chars) const {
|
|
|
|
uint32 charsLen = cStrLen(chars);
|
|
|
|
|
|
|
|
for (int idx = (int)_size - 1; idx >= 0; --idx) {
|
|
|
|
if (!cMemChr(chars, (*this)[idx], charsLen))
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return npos;
|
|
|
|
}
|
|
|
|
|
2023-12-30 13:59:26 +00:00
|
|
|
TEMPLATE void BASESTRING::replace(uint32 pos, uint32 count, const BaseString &str) {
|
|
|
|
replace(pos, count, str, 0, str._size);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::replace(uint32 pos, uint32 count, const value_type *str) {
|
|
|
|
replace(pos, count, str, 0, cStrLen(str));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::replace(iterator begin_, iterator end_, const BaseString &str) {
|
|
|
|
replace(begin_ - _str, end_ - begin_, str._str, 0, str._size);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::replace(iterator begin_, iterator end_, const value_type *str) {
|
|
|
|
replace(begin_ - _str, end_ - begin_, str, 0, cStrLen(str));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::replace(uint32 posOri, uint32 countOri, const BaseString &str,
|
|
|
|
uint32 posDest, uint32 countDest) {
|
|
|
|
replace(posOri, countOri, str._str, posDest, countDest);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::replace(uint32 posOri, uint32 countOri, const value_type *str,
|
|
|
|
uint32 posDest, uint32 countDest) {
|
|
|
|
|
|
|
|
// Prepare string for the replaced text.
|
|
|
|
if (countOri < countDest) {
|
|
|
|
uint32 offset = countDest - countOri; ///< Offset to copy the characters
|
|
|
|
uint32 newSize = _size + offset;
|
2023-12-30 21:38:06 +00:00
|
|
|
uint32 len = _size - posOri;
|
2023-12-30 13:59:26 +00:00
|
|
|
|
|
|
|
ensureCapacity(newSize, true);
|
|
|
|
|
|
|
|
_size = newSize;
|
|
|
|
|
|
|
|
// Push the old characters to the end of the string
|
2023-12-30 21:38:06 +00:00
|
|
|
memmove(_str + posOri + offset, _str + posOri,
|
|
|
|
(len + 1) * sizeof(value_type));
|
2023-12-30 13:59:26 +00:00
|
|
|
|
|
|
|
} else if (countOri > countDest){
|
|
|
|
uint32 offset = countOri - countDest; ///< Number of positions that we have to pull back
|
2023-12-30 21:38:06 +00:00
|
|
|
uint32 len = _size - (posOri + countDest + offset);
|
2023-12-30 13:59:26 +00:00
|
|
|
|
|
|
|
makeUnique();
|
|
|
|
|
|
|
|
// Pull the remainder string back
|
2023-12-30 21:38:06 +00:00
|
|
|
memmove(_str + posOri + countDest, _str + posOri + countDest + offset,
|
|
|
|
(len + 1) * sizeof(value_type));
|
2023-12-30 13:59:26 +00:00
|
|
|
|
|
|
|
_size -= offset;
|
|
|
|
} else {
|
|
|
|
makeUnique();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy the replaced part of the string
|
2023-12-30 21:38:06 +00:00
|
|
|
memmove(_str + posOri, str + posDest, countDest * sizeof(value_type));
|
2023-12-30 13:59:26 +00:00
|
|
|
}
|
|
|
|
|
2023-12-30 22:03:29 +00:00
|
|
|
TEMPLATE void BASESTRING::replace(value_type from, value_type to) {
|
|
|
|
// Don't allow removing trailing \x00
|
|
|
|
if (from == '\x00') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
value_type *next = cMemChr(_str, from, _size);
|
|
|
|
if (!next) {
|
|
|
|
// Nothing to do
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t off = next - _str;
|
|
|
|
makeUnique();
|
|
|
|
|
|
|
|
value_type *end = _str + _size;
|
|
|
|
next = _str + off;
|
|
|
|
while(next) {
|
|
|
|
*next = to;
|
|
|
|
next++;
|
|
|
|
next = cMemChr(next + 1, from, end - next);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE uint64 BASESTRING::asUint64() const {
|
|
|
|
uint64 result = 0;
|
|
|
|
for (uint32 i = 0; i < _size; ++i) {
|
|
|
|
if (_str[i] < '0' || _str[i] > '9') break;
|
|
|
|
result = result * 10L + (_str[i] - '0');
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE uint64 BASESTRING::asUint64Ext() const {
|
|
|
|
uint64 result = 0;
|
|
|
|
uint64 base = 10;
|
|
|
|
uint32 skip = 0;
|
2021-05-04 11:45:03 +03:00
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
if (_size >= 3 && _str[0] == '0' && _str[1] == 'x') {
|
|
|
|
base = 16;
|
|
|
|
skip = 2;
|
|
|
|
} else if (_size >= 2 && _str[0] == '0') {
|
|
|
|
base = 8;
|
|
|
|
skip = 1;
|
|
|
|
} else {
|
|
|
|
base = 10;
|
|
|
|
skip = 0;
|
|
|
|
}
|
|
|
|
for (uint32 i = skip; i < _size; ++i) {
|
|
|
|
char digit = _str[i];
|
|
|
|
uint64 digitval = 17; // sentinel
|
|
|
|
if (digit >= '0' && digit <= '9')
|
|
|
|
digitval = digit - '0';
|
|
|
|
else if (digit >= 'a' && digit <= 'f')
|
|
|
|
digitval = digit - 'a' + 10;
|
|
|
|
else if (digit >= 'A' && digit <= 'F')
|
|
|
|
digitval = digit - 'A' + 10;
|
|
|
|
if (digitval > base)
|
|
|
|
break;
|
|
|
|
result = result * base + digitval;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef SCUMMVM_UTIL
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::wordWrap(const uint32 maxLength) {
|
|
|
|
if (_size < maxLength) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
makeUnique();
|
|
|
|
|
|
|
|
const uint32 kNoSpace = 0xFFFFFFFF;
|
|
|
|
|
|
|
|
uint32 i = 0;
|
|
|
|
while (i < _size) {
|
|
|
|
uint32 lastSpace = kNoSpace;
|
|
|
|
uint32 x = 0;
|
|
|
|
while (i < _size && x <= maxLength) {
|
|
|
|
const char c = _str[i];
|
|
|
|
if (c == '\n') {
|
|
|
|
lastSpace = kNoSpace;
|
|
|
|
x = 0;
|
|
|
|
} else {
|
|
|
|
if (Common::isSpace(c)) {
|
|
|
|
lastSpace = i;
|
|
|
|
}
|
|
|
|
++x;
|
|
|
|
}
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (x > maxLength) {
|
|
|
|
if (lastSpace == kNoSpace) {
|
|
|
|
insertChar('\n', i - 1);
|
|
|
|
} else {
|
|
|
|
setChar('\n', lastSpace);
|
|
|
|
i = lastSpace + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::toLowercase() {
|
2023-09-18 11:28:02 -04:00
|
|
|
toCase(tolower);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::toUppercase() {
|
2023-09-18 11:28:02 -04:00
|
|
|
toCase(toupper);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::toCase(int (*caseChangeFunc)(int)) {
|
|
|
|
uint32 sz = _size;
|
|
|
|
T *buf = _str;
|
|
|
|
|
|
|
|
uint32 i = 0;
|
|
|
|
for ( ; i < sz; ++i) {
|
|
|
|
value_type ch = buf[i];
|
|
|
|
if (ch > 0 && ch < 128) {
|
|
|
|
value_type newCh = static_cast<value_type>(caseChangeFunc(buf[i]));
|
|
|
|
if (ch != newCh) {
|
|
|
|
makeUnique();
|
|
|
|
buf = _str;
|
|
|
|
buf[i] = newCh;
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
}
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
}
|
2023-09-18 11:28:02 -04:00
|
|
|
|
|
|
|
for (; i < sz; ++i) {
|
|
|
|
value_type ch = buf[i];
|
|
|
|
if (ch > 0 && ch < 128)
|
|
|
|
buf[i] = static_cast<value_type>(caseChangeFunc(ch));
|
|
|
|
}
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef SCUMMVM_UTIL
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::trim() {
|
|
|
|
if (_size == 0)
|
|
|
|
return;
|
|
|
|
|
2023-05-08 08:43:35 -04:00
|
|
|
uint numLeading = 0;
|
|
|
|
while (numLeading < _size) {
|
|
|
|
if (isSpace(_str[numLeading]))
|
|
|
|
numLeading++;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint numTrailing = 0;
|
|
|
|
if (numLeading != _size) {
|
|
|
|
while (numTrailing < _size) {
|
|
|
|
if (isSpace(_str[_size - 1 - numTrailing]))
|
|
|
|
numTrailing++;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (numLeading == 0 && numTrailing == 0)
|
|
|
|
return;
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
makeUnique();
|
|
|
|
|
2023-05-08 08:43:35 -04:00
|
|
|
uint newSize = _size - numLeading - numTrailing;
|
2020-10-27 23:22:25 +01:00
|
|
|
|
2023-05-08 08:43:35 -04:00
|
|
|
if (numLeading > 0)
|
|
|
|
memmove(_str, _str + numLeading, newSize);
|
2020-10-27 23:22:25 +01:00
|
|
|
|
2023-05-08 08:43:35 -04:00
|
|
|
_str[newSize] = 0;
|
|
|
|
_size = newSize;
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-08-18 14:20:57 +02:00
|
|
|
TEMPLATE void BASESTRING::append(const value_type *beginP, const value_type *endP) {
|
|
|
|
assert(endP >= beginP);
|
|
|
|
size_t len = endP - beginP;
|
|
|
|
if (len == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Don't test endP as it must be in the same buffer
|
|
|
|
if (pointerInOwnBuffer(beginP)) {
|
|
|
|
assignAppend(BaseString(beginP, endP));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ensureCapacity(_size + len, true);
|
|
|
|
|
|
|
|
memcpy(_str + _size, beginP, len * sizeof(value_type));
|
|
|
|
_size += len;
|
|
|
|
_str[_size] = 0;
|
|
|
|
}
|
|
|
|
|
2023-12-30 16:24:47 +00:00
|
|
|
TEMPLATE void BASESTRING::assignInsert(value_type c, uint32 p) {
|
|
|
|
if (c == 0) {
|
|
|
|
#ifndef SCUMMVM_UTIL
|
|
|
|
warning("Inserting \\0 into String. This is permitted, but can have unwanted consequences. (This warning will be removed later.)");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
ensureCapacity(_size + 1, true);
|
|
|
|
|
|
|
|
memmove(_str + p + 1, _str + p, ((_size - p) + 1) * sizeof(value_type));
|
|
|
|
_str[p] = c;
|
|
|
|
_size++;
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE void BASESTRING::assignAppend(value_type c) {
|
COMMON: Add warning when adding \0 to a String
Some engines do this, and it's not really known what should happen. We
need to decide on a behavior, and stick to that. This warning is to help
find where it happens.
From my own tests, it can happen in at least ADL, Avalanche, Chewy,
Director, Blazing Dragons, MacVenture, MADS, Prince, Stark, Startrek,
and Trecision. Only ADL is known to be broken by the current behavior.
The result of my test can be found at
http://www.update.uu.se/~d91tan/tmp/string.txt
2021-06-29 13:33:17 +02:00
|
|
|
if (c == 0) {
|
2021-06-29 15:39:27 +01:00
|
|
|
#ifndef SCUMMVM_UTIL
|
2021-07-13 10:27:59 +02:00
|
|
|
warning("Adding \\0 to String. This is permitted, but can have unwanted consequences. (This warning will be removed later.)");
|
2021-06-29 15:39:27 +01:00
|
|
|
#endif
|
COMMON: Add warning when adding \0 to a String
Some engines do this, and it's not really known what should happen. We
need to decide on a behavior, and stick to that. This warning is to help
find where it happens.
From my own tests, it can happen in at least ADL, Avalanche, Chewy,
Director, Blazing Dragons, MacVenture, MADS, Prince, Stark, Startrek,
and Trecision. Only ADL is known to be broken by the current behavior.
The result of my test can be found at
http://www.update.uu.se/~d91tan/tmp/string.txt
2021-06-29 13:33:17 +02:00
|
|
|
}
|
2020-10-27 23:22:25 +01:00
|
|
|
ensureCapacity(_size + 1, true);
|
|
|
|
|
|
|
|
_str[_size++] = c;
|
|
|
|
_str[_size] = 0;
|
|
|
|
}
|
|
|
|
|
2023-12-30 16:24:47 +00:00
|
|
|
TEMPLATE void BASESTRING::assignInsert(const BaseString &str, uint32 p) {
|
|
|
|
if (&str == this) {
|
|
|
|
assignInsert(BaseString(str), p);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int len = str._size;
|
|
|
|
if (len > 0) {
|
|
|
|
ensureCapacity(_size + len, true);
|
|
|
|
|
|
|
|
memmove(_str + p + len, _str + p, ((_size - p) + 1) * sizeof(value_type));
|
|
|
|
memcpy(_str + p, str._str, len * sizeof(value_type));
|
|
|
|
_size += len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE void BASESTRING::assignAppend(const BaseString &str) {
|
2020-11-05 23:10:52 +00:00
|
|
|
if (&str == this) {
|
2020-10-27 23:22:25 +01:00
|
|
|
assignAppend(BaseString(str));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int len = str._size;
|
|
|
|
if (len > 0) {
|
|
|
|
ensureCapacity(_size + len, true);
|
|
|
|
|
|
|
|
memcpy(_str + _size, str._str, (len + 1) * sizeof(value_type));
|
|
|
|
_size += len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE bool BASESTRING::pointerInOwnBuffer(const value_type *str) const {
|
|
|
|
//compared pointers must be in the same array or UB
|
|
|
|
//cast to intptr however is IB
|
2024-05-14 08:49:13 +02:00
|
|
|
//which includes comparison of the values
|
2020-10-27 23:22:25 +01:00
|
|
|
uintptr ownBuffStart = (uintptr)_str;
|
|
|
|
uintptr ownBuffEnd = (uintptr)(_str + _size);
|
|
|
|
uintptr candidateAddr = (uintptr)str;
|
|
|
|
return ownBuffStart <= candidateAddr && candidateAddr <= ownBuffEnd;
|
|
|
|
}
|
|
|
|
|
2023-12-30 16:24:47 +00:00
|
|
|
TEMPLATE void BASESTRING::assignInsert(const value_type *str, uint32 p) {
|
|
|
|
if (pointerInOwnBuffer(str)) {
|
|
|
|
assignInsert(BaseString(str), p);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-12-30 21:38:06 +00:00
|
|
|
uint32 len = cStrLen(str);
|
2023-12-30 16:24:47 +00:00
|
|
|
if (len > 0) {
|
|
|
|
ensureCapacity(_size + len, true);
|
|
|
|
|
|
|
|
memmove(_str + p + len, _str + p, ((_size - p) + 1) * sizeof(value_type));
|
|
|
|
memcpy(_str + p, str, len * sizeof(value_type));
|
|
|
|
_size += len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
TEMPLATE void BASESTRING::assignAppend(const value_type *str) {
|
|
|
|
if (pointerInOwnBuffer(str)) {
|
|
|
|
assignAppend(BaseString(str));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-12-30 21:38:06 +00:00
|
|
|
uint32 len = cStrLen(str);
|
2020-10-27 23:22:25 +01:00
|
|
|
if (len > 0) {
|
|
|
|
ensureCapacity(_size + len, true);
|
|
|
|
|
|
|
|
memcpy(_str + _size, str, (len + 1) * sizeof(value_type));
|
|
|
|
_size += len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE void BASESTRING::assign(const value_type *str) {
|
2023-12-30 21:38:06 +00:00
|
|
|
uint32 len = cStrLen(str);
|
2020-10-27 23:22:25 +01:00
|
|
|
ensureCapacity(len, false);
|
|
|
|
_size = len;
|
|
|
|
memmove(_str, str, (len + 1) * sizeof(value_type));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE uint BASESTRING::getUnsignedValue(uint pos) const {
|
|
|
|
const int shift = (sizeof(uint) - sizeof(value_type)) * 8;
|
|
|
|
return ((uint)_str[pos]) << shift >> shift;
|
|
|
|
}
|
|
|
|
|
2023-08-13 10:14:13 -04:00
|
|
|
TEMPLATE uint32 BASESTRING::cStrLen(const value_type *str) {
|
|
|
|
uint32 len = 0;
|
|
|
|
while (str[len])
|
|
|
|
len++;
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2023-12-30 21:38:06 +00:00
|
|
|
TEMPLATE const T *BASESTRING::cMemChr(const value_type *str, value_type c, size_t count) {
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
if (str[i] == c) {
|
|
|
|
return str + i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE T *BASESTRING::cMemChr(value_type *str, value_type c, size_t count) {
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
if (str[i] == c) {
|
|
|
|
return str + i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEMPLATE int BASESTRING::cMemCmp(const value_type* ptr1, const value_type* ptr2, size_t count) {
|
|
|
|
assert(ptr1);
|
|
|
|
assert(ptr2);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
value_type sc = ptr1[i];
|
|
|
|
value_type xc = ptr2[i];
|
|
|
|
if (sc < xc)
|
|
|
|
return -1;
|
|
|
|
else if (sc > xc)
|
|
|
|
return +1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
// Hash function for strings, taken from CPython.
|
|
|
|
TEMPLATE uint BASESTRING::hash() const {
|
2020-11-01 14:58:46 +01:00
|
|
|
uint hashResult = getUnsignedValue(0) << 7;
|
2020-10-27 23:22:25 +01:00
|
|
|
for (uint i = 0; i < _size; i++) {
|
2020-11-01 14:58:46 +01:00
|
|
|
hashResult = (1000003 * hashResult) ^ getUnsignedValue(i);
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
2020-11-01 14:58:46 +01:00
|
|
|
return hashResult ^ _size;
|
2020-10-27 23:22:25 +01:00
|
|
|
}
|
|
|
|
|
2023-08-13 10:20:29 -04:00
|
|
|
template<>
|
|
|
|
uint32 BaseString<char>::cStrLen(const value_type *str) {
|
|
|
|
return static_cast<uint32>(strlen(str));
|
|
|
|
}
|
|
|
|
|
2023-12-30 21:38:06 +00:00
|
|
|
template<>
|
|
|
|
const char *BaseString<char>::cMemChr(const value_type *ptr, value_type c, size_t count) {
|
|
|
|
return static_cast<const char *>(memchr(ptr, c, count));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<>
|
|
|
|
char *BaseString<char>::cMemChr(value_type *ptr, value_type c, size_t count) {
|
|
|
|
return static_cast<char *>(memchr(ptr, c, count));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<>
|
|
|
|
int BaseString<char>::cMemCmp(const value_type* ptr1, const value_type* ptr2, size_t count) {
|
|
|
|
return memcmp(ptr1, ptr2, count);
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:22:25 +01:00
|
|
|
template class BaseString<char>;
|
2023-05-01 23:03:38 +02:00
|
|
|
template class BaseString<uint16>;
|
2020-10-27 23:22:25 +01:00
|
|
|
template class BaseString<u32char_type_t>;
|
|
|
|
|
|
|
|
}
|