mirror of
https://github.com/ultimatepp/ultimatepp.git
synced 2026-05-16 14:16:09 -06:00
This might bring some incompatibilities in the code that expects wchar to be 16 bit, which escpecially involves dealing with Win32 (and to lesser extend MacOS) APIs, so if your application is doing that, please check all instances of WCHAR (UniChar on MacOS) or even wchar especially type casts. To support host APIs, char16 is introduced (but there is no 16-bit String varian). Use ToSystemCharsetW, FromSystemCharsetW to convert texts to Win32 API. - Support of drawing non-BMP characters in GUI - Vastly improved character font replacement code (when drawing characters missing with requested font, replacement font is used) - Last instances of Win32 ANSI calls (those ending with A) are removed - UTF handling routines are refactored and their's naming is unified - RTF is now being able to handle non-BMP characters (RTF is used as clipboard format for RichText) Other minor changes: - fixed TryRealloc issue - improved MemoryCheck - Removed MemoryAlloc48/MemoryFree48 - In theide Background parsing should less often cause delays in the main thread
243 lines
4.9 KiB
C++
243 lines
4.9 KiB
C++
#include "Core.h"
|
|
|
|
namespace Upp {
|
|
|
|
#ifdef CPU_UNALIGNED // Use unaligned access
|
|
|
|
#ifdef CPU_64
|
|
inline bool equal_back_8(const char *a, const char *b, int len)
|
|
{
|
|
while(len > 8) {
|
|
len -= 8;
|
|
if(Peek64le(a + len) != Peek64le(b + len))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
#else
|
|
force_inline bool equal_back_4(const char *a, const char *b, int len)
|
|
{
|
|
while(len > 4) {
|
|
len -= 4;
|
|
if(Peek32le(a + len) != Peek32le(b + len))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
template <int step> // Template parameter to be a constant
|
|
int t_find(const char *ptr, int slen, const char *p, int len, int from)
|
|
{
|
|
ASSERT(from >= 0 && from <= slen);
|
|
int l = slen - len - from;
|
|
if(l < 0)
|
|
return -1;
|
|
const char *s = ptr + from;
|
|
const char *e = s + l;
|
|
if(len <= 8) {
|
|
if(len > 4) {
|
|
len -= 4;
|
|
int32 p0 = Peek32le(p);
|
|
int32 p1 = Peek32le(p + len);
|
|
while(s <= e) {
|
|
if((int32)Peek32le(s) == p0 && (int32)Peek32le(s + len) == p1)
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
}
|
|
else
|
|
if(len == 4) {
|
|
int32 p0 = Peek32le(p);
|
|
while(s <= e) {
|
|
if((int32)Peek32le(s) == p0)
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
}
|
|
else
|
|
if(len == 3) {
|
|
int16 p0 = Peek16le(p);
|
|
char p1 = p[2];
|
|
while(s <= e) {
|
|
if((int16)Peek16le(s) == p0 && s[2] == p1)
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
}
|
|
else
|
|
if(len == 2) {
|
|
int16 p0 = Peek16le(p);
|
|
while(s <= e) {
|
|
if((int16)Peek16le(s) == p0)
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
}
|
|
else
|
|
if(len == 1) {
|
|
char p0 = p[0];
|
|
while(s <= e) {
|
|
if(*s == p0)
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
}
|
|
else
|
|
return from;
|
|
}
|
|
else {
|
|
#ifdef CPU_64
|
|
int64 p0 = Peek64le(p);
|
|
if(len <= 16) {
|
|
len -= 8;
|
|
int64 p1 = Peek64le(p + len);
|
|
while(s <= e) {
|
|
if((int64)Peek64le(s) == p0 && (int64)Peek64le(s + len) == p1)
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
}
|
|
else
|
|
while(s <= e) {
|
|
if((int64)Peek64le(s) == p0 && equal_back_8(s, p, len))
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
#else
|
|
int32 p0 = Peek32le(p);
|
|
while(s <= e) {
|
|
if((int32)Peek32le(s) == p0 && equal_back_4(s, p, len))
|
|
return (int)(s - ptr);
|
|
s += step;
|
|
}
|
|
#endif
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int find(const char *text, int len, const char *needle, int nlen, int from)
|
|
{
|
|
return t_find<1>(text, len, needle, nlen, from);
|
|
}
|
|
|
|
int find(const wchar *text, int len, const wchar *needle, int nlen, int from)
|
|
{
|
|
int q = t_find<sizeof(wchar)>((const char *)text, sizeof(wchar) * len, (const char *)needle, sizeof(wchar) * nlen, sizeof(wchar) * from);
|
|
return q < 0 ? q : q / sizeof(wchar);
|
|
}
|
|
|
|
#else
|
|
|
|
template <class tchar>
|
|
force_inline bool svo_memeq(const tchar *a, const tchar *b, int len)
|
|
{
|
|
if(len > 11)
|
|
return memcmp(a, b, len * sizeof(tchar)) == 0;
|
|
switch(len) {
|
|
case 11:
|
|
if(a[10] != b[10]) return false;
|
|
case 10:
|
|
if(a[9] != b[9]) return false;
|
|
case 9:
|
|
if(a[8] != b[8]) return false;
|
|
case 8:
|
|
if(a[7] != b[7]) return false;
|
|
case 7:
|
|
if(a[6] != b[6]) return false;
|
|
case 6:
|
|
if(a[5] != b[5]) return false;
|
|
case 5:
|
|
if(a[4] != b[4]) return false;
|
|
case 4:
|
|
if(a[3] != b[3]) return false;
|
|
case 3:
|
|
if(a[2] != b[2]) return false;
|
|
case 2:
|
|
if(a[1] != b[1]) return false;
|
|
case 1:
|
|
if(a[0] != b[0]) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <class tchar>
|
|
int t_find(const tchar *ptr, int plen, const tchar *s, int len, int from)
|
|
{
|
|
ASSERT(from >= 0 && from <= plen);
|
|
int l = plen - len - from;
|
|
if(l < 0)
|
|
return -1;
|
|
if(len == 0)
|
|
return from;
|
|
const tchar *p = ptr + from;
|
|
const tchar *e = p + l;
|
|
if(len > 4) {
|
|
tchar s0 = s[0];
|
|
tchar s1 = s[1];
|
|
tchar s2 = s[2];
|
|
tchar sl = s[len - 1];
|
|
int l_1 = len - 1;
|
|
while(p <= e) {
|
|
if(s0 == p[0] && sl == p[l_1] && s1 == p[1] && s2 == p[2] &&
|
|
svo_memeq(s + 3, p + 3, len - 4))
|
|
return (int)(p - ptr);
|
|
p++;
|
|
}
|
|
}
|
|
else
|
|
if(len == 4) {
|
|
tchar s0 = s[0];
|
|
tchar s1 = s[1];
|
|
tchar s2 = s[2];
|
|
tchar s3 = s[3];
|
|
while(p <= e) {
|
|
if(s0 == p[0] && s3 == p[3] && s1 == p[1] && s2 == p[2])
|
|
return (int)(p - ptr);
|
|
p++;
|
|
}
|
|
}
|
|
else
|
|
if(len == 3) {
|
|
tchar s0 = s[0];
|
|
tchar s1 = s[1];
|
|
tchar s2 = s[2];
|
|
while(p <= e) {
|
|
if(s0 == p[0] && s2 == p[2] && s1 == p[1])
|
|
return (int)(p - ptr);
|
|
p++;
|
|
}
|
|
}
|
|
else
|
|
if(len == 2) {
|
|
tchar s0 = s[0];
|
|
tchar s1 = s[1];
|
|
while(p <= e) {
|
|
if(s0 == p[0] && s1 == p[1])
|
|
return (int)(p - ptr);
|
|
p++;
|
|
}
|
|
}
|
|
else {
|
|
tchar s0 = s[0];
|
|
while(p <= e) {
|
|
if(s0 == *p)
|
|
return (int)(p - ptr);
|
|
p++;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int find(const char *text, int len, const char *needle, int nlen, int from)
|
|
{
|
|
return t_find(text, len, needle, nlen, from);
|
|
}
|
|
|
|
int find(const wchar *text, int len, const wchar *needle, int nlen, int from)
|
|
{
|
|
return t_find(text, len, needle, nlen, from);
|
|
}
|
|
#endif
|
|
|
|
}
|