mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-23 14:03:07 +01:00
Slightly reworked the key processing in the hash join algorithm.
This commit is contained in:
parent
1ca0ace21f
commit
15e8bbd4e8
@ -366,12 +366,10 @@ HashJoin::HashJoin(thread_db* tdbb, CompilerScratch* csb, size_t count,
|
||||
dsc desc;
|
||||
(*m_leader.keys)[j]->getDesc(tdbb, csb, &desc);
|
||||
|
||||
USHORT keyLength = desc.dsc_length;
|
||||
USHORT keyLength = desc.isText() ? desc.getStringLength() : desc.dsc_length;
|
||||
|
||||
if (IS_INTL_DATA(&desc))
|
||||
keyLength = INTL_key_length(tdbb, INTL_INDEX_TYPE(&desc), desc.dsc_length);
|
||||
else if (desc.dsc_dtype == dtype_varying)
|
||||
keyLength -= sizeof(USHORT);
|
||||
keyLength = INTL_key_length(tdbb, INTL_INDEX_TYPE(&desc), keyLength);
|
||||
|
||||
m_leader.keyLengths->add(keyLength);
|
||||
m_leader.totalKeyLength += keyLength;
|
||||
@ -394,12 +392,10 @@ HashJoin::HashJoin(thread_db* tdbb, CompilerScratch* csb, size_t count,
|
||||
dsc desc;
|
||||
(*sub.keys)[j]->getDesc(tdbb, csb, &desc);
|
||||
|
||||
USHORT keyLength = desc.dsc_length;
|
||||
USHORT keyLength = desc.isText() ? desc.getStringLength() : desc.dsc_length;
|
||||
|
||||
if (IS_INTL_DATA(&desc))
|
||||
keyLength = INTL_key_length(tdbb, INTL_INDEX_TYPE(&desc), desc.dsc_length);
|
||||
else if (desc.dsc_dtype == dtype_varying)
|
||||
keyLength -= sizeof(USHORT);
|
||||
keyLength = INTL_key_length(tdbb, INTL_INDEX_TYPE(&desc), keyLength);
|
||||
|
||||
sub.keyLengths->add(keyLength);
|
||||
sub.totalKeyLength += keyLength;
|
||||
@ -641,37 +637,35 @@ void HashJoin::computeKeys(thread_db* tdbb, jrd_req* request,
|
||||
{
|
||||
for (size_t i = 0; i < sub.keys->getCount(); i++)
|
||||
{
|
||||
const dsc* const desc = EVL_expr(tdbb, request, (*sub.keys)[i]);
|
||||
dsc* const desc = EVL_expr(tdbb, request, (*sub.keys)[i]);
|
||||
const USHORT keyLength = (*sub.keyLengths)[i];
|
||||
|
||||
if (desc && !(request->req_flags & req_null))
|
||||
{
|
||||
USHORT length = desc->dsc_length;
|
||||
UCHAR* address = desc->dsc_address;
|
||||
|
||||
MoveBuffer buffer;
|
||||
|
||||
if (IS_INTL_DATA(desc))
|
||||
if (desc->isText())
|
||||
{
|
||||
// Convert the INTL string into the binary comparable form
|
||||
dsc to;
|
||||
to.makeText(keyLength, ttype_binary, keyBuffer);
|
||||
|
||||
address = buffer.getBuffer(keyLength);
|
||||
|
||||
dsc temp;
|
||||
temp.makeText(keyLength, ttype_sort_key, address);
|
||||
|
||||
length = INTL_string_to_key(tdbb, INTL_INDEX_TYPE(desc),
|
||||
desc, &temp, INTL_KEY_UNIQUE);
|
||||
if (IS_INTL_DATA(desc))
|
||||
{
|
||||
// Convert the INTL string into the binary comparable form
|
||||
INTL_string_to_key(tdbb, INTL_INDEX_TYPE(desc),
|
||||
desc, &to, INTL_KEY_UNIQUE);
|
||||
}
|
||||
else
|
||||
{
|
||||
// This call ensures that the padding bytes are appended
|
||||
MOV_move(tdbb, desc, &to);
|
||||
}
|
||||
}
|
||||
else if (desc->dsc_dtype == dtype_varying)
|
||||
else
|
||||
{
|
||||
length -= sizeof(USHORT);
|
||||
address += sizeof(USHORT);
|
||||
// We don't enforce proper alignments inside the key buffer,
|
||||
// so use plain byte copying instead of MOV_move() to avoid bus errors
|
||||
fb_assert(keyLength == desc->dsc_length);
|
||||
memcpy(keyBuffer, desc->dsc_address, keyLength);
|
||||
}
|
||||
|
||||
fb_assert(length <= keyLength);
|
||||
|
||||
memcpy(keyBuffer, address, length);
|
||||
}
|
||||
|
||||
keyBuffer += keyLength;
|
||||
|
Loading…
Reference in New Issue
Block a user