Commit e85df7fe authored by Vladislav Vaintroub's avatar Vladislav Vaintroub

MDEV-19702 Refactor Bitmap<N> to be based on ulonglong, not on uint32

Removed Field_map, since it was used only in a single function.

Fixed is_indexed_agg_distinct(), since it relied on initialization of
Bitmap in constructor.

Fixes MDEV-25888 in 10.4
parent 0b9a59bb
...@@ -27,10 +27,36 @@ ...@@ -27,10 +27,36 @@
#include <my_bitmap.h> #include <my_bitmap.h>
#include <my_bit.h> #include <my_bit.h>
/* An iterator to quickly walk over bits in unlonglong bitmap. */
class Table_map_iterator
{
ulonglong bmp;
uint no;
public:
Table_map_iterator(ulonglong t) : bmp(t), no(0) {}
int next_bit()
{
static const char last_bit[16] = { 32, 0, 1, 0,
2, 0, 1, 0,
3, 0, 1, 0,
2, 0, 1, 0 };
uint bit;
while ((bit= last_bit[bmp & 0xF]) == 32)
{
no += 4;
bmp= bmp >> 4;
if (!bmp)
return BITMAP_END;
}
bmp &= ~(1LL << bit);
return no + bit;
}
int operator++(int) { return next_bit(); }
enum { BITMAP_END= 64 };
};
template <uint width> class Bitmap template <uint width> class Bitmap
{ {
/* /*
Workaround GCC optimizer bug (generating SSE instuctions on unaligned data) Workaround GCC optimizer bug (generating SSE instuctions on unaligned data)
*/ */
...@@ -43,12 +69,38 @@ template <uint width> class Bitmap ...@@ -43,12 +69,38 @@ template <uint width> class Bitmap
#pragma GCC target ("no-sse") #pragma GCC target ("no-sse")
#endif #endif
uint32 buffer[(width + 31) / 32]; private:
public: static const int BITS_PER_ELEMENT= sizeof(ulonglong) * 8;
Bitmap() static const int ARRAY_ELEMENTS= (width + BITS_PER_ELEMENT - 1) / BITS_PER_ELEMENT;
static const ulonglong ALL_BITS_SET= ULLONG_MAX;
ulonglong buffer[ARRAY_ELEMENTS];
uint bit_index(uint n) const
{
DBUG_ASSERT(n < width);
return ARRAY_ELEMENTS == 1 ? 0 : n / BITS_PER_ELEMENT;
}
ulonglong bit_mask(uint n) const
{ {
clear_all(); DBUG_ASSERT(n < width);
return ARRAY_ELEMENTS == 1 ? 1ULL << n : 1ULL << (n % BITS_PER_ELEMENT);
} }
ulonglong last_element_mask(int n) const
{
DBUG_ASSERT(n % BITS_PER_ELEMENT != 0);
return bit_mask(n) - 1;
}
public:
/*
The default constructor does nothing.
The caller is supposed to either zero the memory
or to call set_all()/clear_all()/set_prefix()
to initialize bitmap.
*/
Bitmap() { }
explicit Bitmap(uint prefix) explicit Bitmap(uint prefix)
{ {
set_prefix(prefix); set_prefix(prefix);
...@@ -57,50 +109,76 @@ template <uint width> class Bitmap ...@@ -57,50 +109,76 @@ template <uint width> class Bitmap
{ {
set_prefix(prefix); set_prefix(prefix);
} }
uint length() const uint length() const
{ {
return width; return width;
} }
void set_bit(uint n) void set_bit(uint n)
{ {
DBUG_ASSERT(n < width); buffer[bit_index(n)] |= bit_mask(n);
((uchar*)buffer)[n / 8] |= (1 << (n & 7));
} }
void clear_bit(uint n) void clear_bit(uint n)
{ {
DBUG_ASSERT(n < width); buffer[bit_index(n)] &= ~bit_mask(n);
((uchar*)buffer)[n / 8] &= ~(1 << (n & 7)); }
bool is_set(uint n) const
{
return buffer[bit_index(n)] & bit_mask(n);
} }
void set_prefix(uint prefix_size) void set_prefix(uint prefix_size)
{ {
set_if_smaller(prefix_size, width); set_if_smaller(prefix_size, width);
uint prefix_bytes, prefix_bits, d;
uchar* m = (uchar*)buffer;
if ((prefix_bytes = prefix_size / 8)) size_t idx= prefix_size / BITS_PER_ELEMENT;
memset(m, 0xff, prefix_bytes);
m += prefix_bytes; for (size_t i= 0; i < idx; i++)
if ((prefix_bits = prefix_size & 7)) buffer[i]= ALL_BITS_SET;
{
*(m++) = (1 << prefix_bits) - 1; if (prefix_size % BITS_PER_ELEMENT)
// As the prefix bits are set, lets count this byte too as a prefix byte. buffer[idx++]= last_element_mask(prefix_size);
prefix_bytes++;
} for (size_t i= idx; i < ARRAY_ELEMENTS; i++)
if ((d = (width + 7) / 8 - prefix_bytes)) buffer[i]= 0;
memset(m, 0, d); }
bool is_prefix(uint prefix_size) const
{
DBUG_ASSERT(prefix_size <= width);
size_t idx= prefix_size / BITS_PER_ELEMENT;
for (size_t i= 0; i < idx; i++)
if (buffer[i] != ALL_BITS_SET)
return false;
if (prefix_size % BITS_PER_ELEMENT)
if (buffer[idx++] != last_element_mask(prefix_size))
return false;
for (size_t i= idx; i < ARRAY_ELEMENTS; i++)
if (buffer[i] != 0)
return false;
return true;
} }
void set_all() void set_all()
{ {
set_prefix(width); if (width % BITS_PER_ELEMENT)
set_prefix(width);
else if (ARRAY_ELEMENTS > 1)
memset(buffer, 0xff, sizeof(buffer));
else
buffer[0] = ALL_BITS_SET;
} }
void clear_all() void clear_all()
{ {
memset(buffer, 0x00, sizeof(buffer)); if (ARRAY_ELEMENTS > 1)
memset(buffer, 0, sizeof(buffer));
else
buffer[0]= 0;
} }
void intersect(Bitmap & map2) void intersect(const Bitmap& map2)
{ {
for (uint i = 0; i < array_elements(buffer); i++) for (size_t i= 0; i < ARRAY_ELEMENTS; i++)
buffer[i] &= map2.buffer[i]; buffer[i] &= map2.buffer[i];
} }
...@@ -112,27 +190,13 @@ template <uint width> class Bitmap ...@@ -112,27 +190,13 @@ template <uint width> class Bitmap
*/ */
void intersect_and_pad(ulonglong map2buff, bool pad_with_ones) void intersect_and_pad(ulonglong map2buff, bool pad_with_ones)
{ {
compile_time_assert(sizeof(ulonglong) == 8); buffer[0] &= map2buff;
uint32 tmp[2];
int8store(tmp, map2buff);
buffer[0] &= tmp[0]; for (size_t i= 1; i < ARRAY_ELEMENTS; i++)
if (array_elements(buffer) > 1) buffer[i]= pad_with_ones ? ALL_BITS_SET : 0;
buffer[1] &= tmp[1];
if (array_elements(buffer) <= 2)
return;
if (pad_with_ones)
{
memset((char*)buffer + 8, 0xff , sizeof(buffer) - 8);
if (width != sizeof(buffer) * 8)
{
((uchar*)buffer)[sizeof(buffer)-1] = last_byte_mask(width);
}
}
else
memset((char*)buffer + 8, 0 , sizeof(buffer) - 8);
if (ARRAY_ELEMENTS > 1 && (width % BITS_PER_ELEMENT) && pad_with_ones)
buffer[ARRAY_ELEMENTS - 1]= last_element_mask(width);
} }
public: public:
...@@ -140,141 +204,110 @@ template <uint width> class Bitmap ...@@ -140,141 +204,110 @@ template <uint width> class Bitmap
{ {
intersect_and_pad(map2buff, 0); intersect_and_pad(map2buff, 0);
} }
/* Use highest bit for all bits above sizeof(ulonglong)*8. */ /* Use highest bit for all bits above first element. */
void intersect_extended(ulonglong map2buff) void intersect_extended(ulonglong map2buff)
{ {
intersect_and_pad(map2buff, (map2buff & (1ULL << 63))); intersect_and_pad(map2buff, (map2buff & (1ULL << 63)));
} }
void subtract(Bitmap & map2) void subtract(const Bitmap& map2)
{ {
for (size_t i = 0; i < array_elements(buffer); i++) for (size_t i= 0; i < ARRAY_ELEMENTS; i++)
buffer[i] &= ~(map2.buffer[i]); buffer[i] &= ~(map2.buffer[i]);
} }
void merge(Bitmap & map2) void merge(const Bitmap& map2)
{ {
for (size_t i = 0; i < array_elements(buffer); i++) for (size_t i= 0; i < ARRAY_ELEMENTS; i++)
buffer[i] |= map2.buffer[i]; buffer[i] |= map2.buffer[i];
} }
bool is_set(uint n) const
{
DBUG_ASSERT(n < width);
return ((uchar*)buffer)[n / 8] & (1 << (n & 7));
}
bool is_prefix(uint prefix_size) const
{
uint prefix_mask = last_byte_mask(prefix_size);
uchar* m = (uchar*)buffer;
uchar* end_prefix = m + (prefix_size - 1) / 8;
uchar* end;
DBUG_ASSERT(prefix_size <= width);
/* Empty prefix is always true */
if (!prefix_size)
return true;
while (m < end_prefix)
if (*m++ != 0xff)
return false;
end = ((uchar*)buffer) + (width + 7) / 8 - 1;
if (m == end)
return ((*m & last_byte_mask(width)) == prefix_mask);
if (*m != prefix_mask)
return false;
while (++m < end)
if (*m != 0)
return false;
return ((*m & last_byte_mask(width)) == 0);
}
bool is_clear_all() const bool is_clear_all() const
{ {
for (size_t i= 0; i < array_elements(buffer); i++) for (size_t i= 0; i < ARRAY_ELEMENTS; i++)
if (buffer[i]) if (buffer[i])
return false; return false;
return true; return true;
} }
bool is_set_all() const bool is_subset(const Bitmap& map2) const
{
if (width == sizeof(buffer) * 8)
{
for (size_t i = 0; i < array_elements(buffer); i++)
if (buffer[i] != 0xFFFFFFFFU)
return false;
return true;
}
else
return is_prefix(width);
}
bool is_subset(const Bitmap & map2) const
{ {
for (size_t i= 0; i < array_elements(buffer); i++) for (size_t i= 0; i < ARRAY_ELEMENTS; i++)
if (buffer[i] & ~(map2.buffer[i])) if (buffer[i] & ~(map2.buffer[i]))
return false; return false;
return true; return true;
} }
bool is_overlapping(const Bitmap & map2) const bool is_overlapping(const Bitmap& map2) const
{ {
for (size_t i = 0; i < array_elements(buffer); i++) for (size_t i= 0; i < ARRAY_ELEMENTS; i++)
if (buffer[i] & map2.buffer[i]) if (buffer[i] & map2.buffer[i])
return true; return true;
return false; return false;
} }
bool operator==(const Bitmap & map2) const bool operator==(const Bitmap& map2) const
{ {
return memcmp(buffer, map2.buffer, sizeof(buffer)) == 0; if (ARRAY_ELEMENTS > 1)
return !memcmp(buffer,map2.buffer,sizeof(buffer));
return buffer[0] == map2.buffer[0];
} }
bool operator!=(const Bitmap & map2) const bool operator!=(const Bitmap& map2) const
{ {
return !(*this == map2); return !(*this == map2);
} }
/*
Print hexadecimal representation of bitmap.
Truncate trailing zeros.
*/
char *print(char *buf) const char *print(char *buf) const
{ {
char *s=buf; size_t last; /*index of the last non-zero element, or 0. */
const uchar *e=(uchar *)buffer, *b=e+sizeof(buffer)-1;
while (!*b && b>e) for (last= ARRAY_ELEMENTS - 1; last && !buffer[last]; last--){}
b--;
if ((*s=_dig_vec_upper[*b >> 4]) != '0') const int HEX_DIGITS_PER_ELEMENT= BITS_PER_ELEMENT / 4;
s++; for (size_t i= 0; i < last; i++)
*s++=_dig_vec_upper[*b & 15];
while (--b>=e)
{ {
*s++=_dig_vec_upper[*b >> 4]; ulonglong num = buffer[i];
*s++=_dig_vec_upper[*b & 15]; uint shift = BITS_PER_ELEMENT - 4;
size_t pos= i * HEX_DIGITS_PER_ELEMENT;
for (size_t j= 0; j < HEX_DIGITS_PER_ELEMENT; j++)
{
buf[pos + j]= _dig_vec_upper[(num >> shift) & 0xf];
shift += 4;
}
} }
*s=0; longlong2str(buffer[last], buf, 16);
return buf; return buf;
} }
ulonglong to_ulonglong() const ulonglong to_ulonglong() const
{ {
DBUG_ASSERT(sizeof(buffer) >= 4); return buffer[0];
uchar *b=(uchar *)buffer;
if (sizeof(buffer) >= 8)
return uint8korr(b);
return (ulonglong) uint4korr(b);
} }
uint bits_set() uint bits_set()
{ {
uint res = 0; uint res= 0;
for (size_t i = 0; i < array_elements(buffer); i++) for (size_t i= 0; i < ARRAY_ELEMENTS; i++)
res += my_count_bits_uint32(buffer[i]); res += my_count_bits(buffer[i]);
return res; return res;
} }
class Iterator class Iterator
{ {
Bitmap &map; const Bitmap& map;
uint no; uint offset;
Table_map_iterator tmi;
public: public:
Iterator(Bitmap<width> &map2): map(map2), no(0) {} Iterator(const Bitmap<width>& map2) : map(map2), offset(0), tmi(map2.buffer[0]) {}
int operator++(int) { int operator++(int)
if (no == width) return BITMAP_END; {
while (!map.is_set(no)) for (;;)
{ {
if ((++no) == width) return BITMAP_END; int nextbit= tmi++;
if (nextbit != Table_map_iterator::BITMAP_END)
return offset + nextbit;
if (offset + BITS_PER_ELEMENT >= map.length())
return BITMAP_END;
offset += BITS_PER_ELEMENT;
tmi= Table_map_iterator(map.buffer[offset / BITS_PER_ELEMENT]);
} }
return no++;
} }
enum { BITMAP_END = width }; enum { BITMAP_END = width };
}; };
...@@ -283,98 +316,8 @@ template <uint width> class Bitmap ...@@ -283,98 +316,8 @@ template <uint width> class Bitmap
#pragma GCC pop_options #pragma GCC pop_options
#undef NEED_GCC_NO_SSE_WORKAROUND #undef NEED_GCC_NO_SSE_WORKAROUND
#endif #endif
};
/* An iterator to quickly walk over bits in ulonglong bitmap. */
class Table_map_iterator
{
ulonglong bmp;
uint no;
public:
Table_map_iterator(ulonglong t) : bmp(t), no(0) {}
uint next_bit()
{
static const uchar last_bit[16]= {32, 0, 1, 0,
2, 0, 1, 0,
3, 0, 1, 0,
2, 0, 1, 0};
uint bit;
while ((bit= last_bit[bmp & 0xF]) == 32)
{
no += 4;
bmp= bmp >> 4;
if (!bmp)
return BITMAP_END;
}
bmp &= ~(1ULL << bit);
return no + bit;
}
uint operator++(int) { return next_bit(); }
enum { BITMAP_END= 64 };
}; };
template <> class Bitmap<64> typedef Bitmap<MAX_INDEXES> key_map; /* Used for finding keys */
{
ulonglong map;
public:
Bitmap<64>() { }
explicit Bitmap<64>(uint prefix_to_set) { set_prefix(prefix_to_set); }
void init(uint prefix_to_set) { set_prefix(prefix_to_set); }
uint length() const { return 64; }
void set_bit(uint n) { map|= ((ulonglong)1) << n; }
void clear_bit(uint n) { map&= ~(((ulonglong)1) << n); }
void set_prefix(uint n)
{
if (n >= length())
set_all();
else
map= (((ulonglong)1) << n)-1;
}
void set_all() { map=~(ulonglong)0; }
void clear_all() { map=(ulonglong)0; }
void intersect(Bitmap<64>& map2) { map&= map2.map; }
void intersect(ulonglong map2) { map&= map2; }
void intersect_extended(ulonglong map2) { map&= map2; }
void subtract(Bitmap<64>& map2) { map&= ~map2.map; }
void merge(Bitmap<64>& map2) { map|= map2.map; }
bool is_set(uint n) const { return MY_TEST(map & (((ulonglong) 1) << n)); }
bool is_prefix(uint n) const { return map == (((ulonglong)1) << n)-1; }
bool is_clear_all() const { return map == (ulonglong)0; }
bool is_set_all() const { return map == ~(ulonglong)0; }
bool is_subset(const Bitmap<64>& map2) const { return !(map & ~map2.map); }
bool is_overlapping(const Bitmap<64>& map2) const { return (map & map2.map)!= 0; }
bool operator==(const Bitmap<64>& map2) const { return map == map2.map; }
char *print(char *buf) const {
longlong2str(longlong(map), buf, 16);
return buf;
}
ulonglong to_ulonglong() const { return map; }
class Iterator : public Table_map_iterator
{
public:
Iterator(Bitmap<64> &map2) : Table_map_iterator(map2.map) {}
};
uint bits_set()
{
//TODO: use my_count_bits()
uint res= 0, i= 0;
for (; i < 64 ; i++)
{
if (map & ((ulonglong)1<<i))
res++;
}
return res;
}
};
#if MAX_INDEXES <= 64
typedef Bitmap<64> key_map; /* Used for finding keys */
#elif MAX_INDEXES > 128
#error "MAX_INDEXES values greater than 128 is not supported."
#else
typedef Bitmap<((MAX_INDEXES+7)/8*8)> key_map; /* Used for finding keys */
#endif
#endif /* SQL_BITMAP_INCLUDED */ #endif /* SQL_BITMAP_INCLUDED */
...@@ -7046,7 +7046,6 @@ void optimize_keyuse(JOIN *join, DYNAMIC_ARRAY *keyuse_array) ...@@ -7046,7 +7046,6 @@ void optimize_keyuse(JOIN *join, DYNAMIC_ARRAY *keyuse_array)
} }
} }
/** /**
Check for the presence of AGGFN(DISTINCT a) queries that may be subject Check for the presence of AGGFN(DISTINCT a) queries that may be subject
to loose index scan. to loose index scan.
...@@ -7084,7 +7083,6 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args) ...@@ -7084,7 +7083,6 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args)
{ {
Item_sum **sum_item_ptr; Item_sum **sum_item_ptr;
bool result= false; bool result= false;
Field_map first_aggdistinct_fields;
if (join->table_count != 1 || /* reference more than 1 table */ if (join->table_count != 1 || /* reference more than 1 table */
join->select_distinct || /* or a DISTINCT */ join->select_distinct || /* or a DISTINCT */
...@@ -7094,10 +7092,11 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args) ...@@ -7094,10 +7092,11 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args)
if (join->make_sum_func_list(join->all_fields, join->fields_list, true)) if (join->make_sum_func_list(join->all_fields, join->fields_list, true))
return false; return false;
Bitmap<MAX_FIELDS> first_aggdistinct_fields;
bool first_aggdistinct_fields_initialized= false;
for (sum_item_ptr= join->sum_funcs; *sum_item_ptr; sum_item_ptr++) for (sum_item_ptr= join->sum_funcs; *sum_item_ptr; sum_item_ptr++)
{ {
Item_sum *sum_item= *sum_item_ptr; Item_sum *sum_item= *sum_item_ptr;
Field_map cur_aggdistinct_fields;
Item *expr; Item *expr;
/* aggregate is not AGGFN(DISTINCT) or more than 1 argument to it */ /* aggregate is not AGGFN(DISTINCT) or more than 1 argument to it */
switch (sum_item->sum_func()) switch (sum_item->sum_func())
...@@ -7120,6 +7119,8 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args) ...@@ -7120,6 +7119,8 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args)
We don't worry about duplicates as these will be sorted out later in We don't worry about duplicates as these will be sorted out later in
get_best_group_min_max get_best_group_min_max
*/ */
Bitmap<MAX_FIELDS> cur_aggdistinct_fields;
cur_aggdistinct_fields.clear_all();
for (uint i= 0; i < sum_item->get_arg_count(); i++) for (uint i= 0; i < sum_item->get_arg_count(); i++)
{ {
expr= sum_item->get_arg(i); expr= sum_item->get_arg(i);
...@@ -7138,8 +7139,11 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args) ...@@ -7138,8 +7139,11 @@ is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args)
If there are multiple aggregate functions, make sure that they all If there are multiple aggregate functions, make sure that they all
refer to exactly the same set of columns. refer to exactly the same set of columns.
*/ */
if (first_aggdistinct_fields.is_clear_all()) if (!first_aggdistinct_fields_initialized)
first_aggdistinct_fields.merge(cur_aggdistinct_fields); {
first_aggdistinct_fields= cur_aggdistinct_fields;
first_aggdistinct_fields_initialized=true;
}
else if (first_aggdistinct_fields != cur_aggdistinct_fields) else if (first_aggdistinct_fields != cur_aggdistinct_fields)
return false; return false;
} }
......
...@@ -1193,9 +1193,6 @@ struct st_cond_statistic; ...@@ -1193,9 +1193,6 @@ struct st_cond_statistic;
#define CHECK_ROW_FOR_NULLS_TO_REJECT (1 << 0) #define CHECK_ROW_FOR_NULLS_TO_REJECT (1 << 0)
#define REJECT_ROW_DUE_TO_NULL_FIELDS (1 << 1) #define REJECT_ROW_DUE_TO_NULL_FIELDS (1 << 1)
/* Bitmap of table's fields */
typedef Bitmap<MAX_FIELDS> Field_map;
class SplM_opt_info; class SplM_opt_info;
struct vers_select_conds_t; struct vers_select_conds_t;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment