8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-23 18:03:04 +01:00

Improvement #6872 - Indexed STARTING WITH execution is very slow with UNICODE collation.

This commit is contained in:
Adriano dos Santos Fernandes 2021-06-25 15:37:25 -03:00
parent 5131752ce3
commit b9d5ac0fee
2 changed files with 44 additions and 29 deletions

View File

@ -1471,10 +1471,6 @@ UnicodeUtil::Utf16Collation* UnicodeUtil::Utf16Collation::create(
icu->ucolSetAttribute(compareCollator, UCOL_STRENGTH, UCOL_SECONDARY, &status); icu->ucolSetAttribute(compareCollator, UCOL_STRENGTH, UCOL_SECONDARY, &status);
} }
USet* contractions = icu->usetOpen(0, 0);
// status not verified here.
icu->ucolGetContractions(partialCollator, contractions, &status);
Utf16Collation* obj = FB_NEW Utf16Collation(); Utf16Collation* obj = FB_NEW Utf16Collation();
obj->icu = icu; obj->icu = icu;
obj->tt = tt; obj->tt = tt;
@ -1482,9 +1478,39 @@ UnicodeUtil::Utf16Collation* UnicodeUtil::Utf16Collation::create(
obj->compareCollator = compareCollator; obj->compareCollator = compareCollator;
obj->partialCollator = partialCollator; obj->partialCollator = partialCollator;
obj->sortCollator = sortCollator; obj->sortCollator = sortCollator;
obj->contractions = contractions;
obj->contractionsCount = icu->usetGetItemCount(contractions);
obj->numericSort = isNumericSort; obj->numericSort = isNumericSort;
obj->maxContractionsPrefixLength = 0;
USet* contractions = icu->usetOpen(1, 0);
// status not verified here.
icu->ucolGetContractions(partialCollator, contractions, &status);
int contractionsCount = icu->usetGetItemCount(contractions);
for (int contractionIndex = 0; contractionIndex < contractionsCount; ++contractionIndex)
{
UChar str[10];
UChar32 start, end;
status = U_ZERO_ERROR;
int len = icu->usetGetItem(contractions, contractionIndex, &start, &end, str, sizeof(str), &status);
if (len >= 2)
{
obj->maxContractionsPrefixLength = len - 1 > obj->maxContractionsPrefixLength ?
len - 1 : obj->maxContractionsPrefixLength;
for (int currentLen = 1; currentLen < len; ++currentLen)
{
string s(reinterpret_cast<const char*>(str), currentLen * 2);
if (!obj->contractionsPrefix.exist(s))
obj->contractionsPrefix.push(s);
}
}
}
icu->usetClose(contractions);
return obj; return obj;
} }
@ -1492,8 +1518,6 @@ UnicodeUtil::Utf16Collation* UnicodeUtil::Utf16Collation::create(
UnicodeUtil::Utf16Collation::~Utf16Collation() UnicodeUtil::Utf16Collation::~Utf16Collation()
{ {
icu->usetClose(contractions);
icu->ucolClose(compareCollator); icu->ucolClose(compareCollator);
icu->ucolClose(partialCollator); icu->ucolClose(partialCollator);
icu->ucolClose(sortCollator); icu->ucolClose(sortCollator);
@ -1544,30 +1568,16 @@ USHORT UnicodeUtil::Utf16Collation::stringToKey(USHORT srcLen, const USHORT* src
switch (key_type) switch (key_type)
{ {
case INTL_KEY_PARTIAL: case INTL_KEY_PARTIAL:
{
coll = partialCollator; coll = partialCollator;
// Remove last bytes of key if they are start of a contraction // Remove last bytes of key if they are start of a contraction
// to correctly find in the index. // to correctly find in the index.
ConversionICU& cIcu(getConversionICU());
for (int i = 0; i < contractionsCount; ++i) for (int i = MIN(maxContractionsPrefixLength, srcLenLong); i > 0; --i)
{ {
UChar str[10]; if (contractionsPrefix.exist(string(reinterpret_cast<const char*>(src + srcLenLong - i), i * 2)))
UErrorCode status = U_ZERO_ERROR;
int len = icu->usetGetItem(contractions, i, NULL, NULL, str, sizeof(str), &status);
if (len < 0)
fatal_exception::raiseFmt("uset_getItem() error %d", status);
if (unsigned(len) > srcLenLong) // safe cast - sign checked
len = srcLenLong;
else
--len;
// safe cast - alignment not changed
if (cIcu.u_strCompare(str, len,
reinterpret_cast<const UChar*>(src) + srcLenLong - len, len, true) == 0)
{ {
srcLenLong -= len; srcLenLong -= i;
break; break;
} }
} }
@ -1588,7 +1598,6 @@ USHORT UnicodeUtil::Utf16Collation::stringToKey(USHORT srcLen, const USHORT* src
} }
break; break;
}
case INTL_KEY_UNIQUE: case INTL_KEY_UNIQUE:
coll = compareCollator; coll = compareCollator;

View File

@ -31,6 +31,7 @@
#include "../common/IntlUtil.h" #include "../common/IntlUtil.h"
#include "../common/os/mod_loader.h" #include "../common/os/mod_loader.h"
#include "../common/classes/fb_string.h" #include "../common/classes/fb_string.h"
#include "../common/classes/objects_array.h"
#include <unicode/ucnv.h> #include <unicode/ucnv.h>
#include <unicode/ucal.h> #include <unicode/ucal.h>
@ -184,6 +185,11 @@ public:
Firebird::IntlUtil::SpecificAttributesMap& specificAttributes, Firebird::IntlUtil::SpecificAttributesMap& specificAttributes,
const Firebird::string& configInfo); const Firebird::string& configInfo);
Utf16Collation()
: contractionsPrefix(*getDefaultMemoryPool())
{
}
~Utf16Collation(); ~Utf16Collation();
USHORT keyLength(USHORT len) const; USHORT keyLength(USHORT len) const;
@ -206,8 +212,8 @@ public:
UCollator* compareCollator; UCollator* compareCollator;
UCollator* partialCollator; UCollator* partialCollator;
UCollator* sortCollator; UCollator* sortCollator;
USet* contractions; Firebird::SortedObjectsArray<Firebird::string> contractionsPrefix; // UTF-16 string
int contractionsCount; unsigned maxContractionsPrefixLength; // number of characters
bool numericSort; bool numericSort;
}; };