Commit 85d338cc authored by Rich Prohaska's avatar Rich Prohaska

#222 filter out degenerate key range in tokudb::record_in_range

parent 53cb5329
...@@ -7329,23 +7329,18 @@ double ha_tokudb::index_only_read_time(uint keynr, double records) { ...@@ -7329,23 +7329,18 @@ double ha_tokudb::index_only_read_time(uint keynr, double records) {
// //
ha_rows ha_tokudb::records_in_range(uint keynr, key_range* start_key, key_range* end_key) { ha_rows ha_tokudb::records_in_range(uint keynr, key_range* start_key, key_range* end_key) {
TOKUDB_HANDLER_DBUG_ENTER(""); TOKUDB_HANDLER_DBUG_ENTER("");
DBT *pleft_key = NULL, *pright_key = NULL; DBT *pleft_key, *pright_key;
DBT left_key, right_key; DBT left_key, right_key;
ha_rows ret_val = HA_TOKUDB_RANGE_COUNT; ha_rows ret_val = HA_TOKUDB_RANGE_COUNT;
DB *kfile = share->key_file[keynr]; DB *kfile = share->key_file[keynr];
uint64_t less, equal1, middle, equal2, greater; uint64_t rows = 0;
uint64_t rows;
bool is_exact;
int error; int error;
uchar inf_byte;
//
// get start_rows and end_rows values so that we can estimate range // get start_rows and end_rows values so that we can estimate range
// when calling key_range64, the only value we can trust is the value for less // when calling key_range64, the only value we can trust is the value for less
// The reason is that the key being passed in may be a prefix of keys in the DB // The reason is that the key being passed in may be a prefix of keys in the DB
// As a result, equal may be 0 and greater may actually be equal+greater // As a result, equal may be 0 and greater may actually be equal+greater
// So, we call key_range64 on the key, and the key that is after it. // So, we call key_range64 on the key, and the key that is after it.
//
if (!start_key && !end_key) { if (!start_key && !end_key) {
error = estimate_num_rows(kfile, &rows, transaction); error = estimate_num_rows(kfile, &rows, transaction);
if (error) { if (error) {
...@@ -7356,54 +7351,38 @@ ha_rows ha_tokudb::records_in_range(uint keynr, key_range* start_key, key_range* ...@@ -7356,54 +7351,38 @@ ha_rows ha_tokudb::records_in_range(uint keynr, key_range* start_key, key_range*
goto cleanup; goto cleanup;
} }
if (start_key) { if (start_key) {
inf_byte = (start_key->flag == HA_READ_KEY_EXACT) ? uchar inf_byte = (start_key->flag == HA_READ_KEY_EXACT) ? COL_NEG_INF : COL_POS_INF;
COL_NEG_INF : COL_POS_INF; pack_key(&left_key, keynr, key_buff, start_key->key, start_key->length, inf_byte);
pack_key(
&left_key,
keynr,
key_buff,
start_key->key,
start_key->length,
inf_byte
);
pleft_key = &left_key; pleft_key = &left_key;
} else {
pleft_key = NULL;
} }
if (end_key) { if (end_key) {
inf_byte = (end_key->flag == HA_READ_BEFORE_KEY) ? uchar inf_byte = (end_key->flag == HA_READ_BEFORE_KEY) ? COL_NEG_INF : COL_POS_INF;
COL_NEG_INF : COL_POS_INF; pack_key(&right_key, keynr, key_buff2, end_key->key, end_key->length, inf_byte);
pack_key(
&right_key,
keynr,
key_buff2,
end_key->key,
end_key->length,
inf_byte
);
pright_key = &right_key; pright_key = &right_key;
} else {
pright_key = NULL;
} }
error = kfile->keys_range64( // keys_range64 can not handle a degenerate range (left_key > right_key), so we filter here
kfile, if (pleft_key && pright_key && tokudb_cmp_dbt_key(kfile, pleft_key, pright_key) > 0) {
transaction, rows = 0;
pleft_key, } else {
pright_key, uint64_t less, equal1, middle, equal2, greater;
&less, bool is_exact;
&equal1, error = kfile->keys_range64(kfile, transaction, pleft_key, pright_key,
&middle, &less, &equal1, &middle, &equal2, &greater, &is_exact);
&equal2, if (error) {
&greater, ret_val = HA_TOKUDB_RANGE_COUNT;
&is_exact goto cleanup;
); }
if (error) { rows = middle;
ret_val = HA_TOKUDB_RANGE_COUNT;
goto cleanup;
} }
rows = middle;
//
// MySQL thinks a return value of 0 means there are exactly 0 rows // MySQL thinks a return value of 0 means there are exactly 0 rows
// Therefore, always return non-zero so this assumption is not made // Therefore, always return non-zero so this assumption is not made
//
ret_val = (ha_rows) (rows <= 1 ? 1 : rows); ret_val = (ha_rows) (rows <= 1 ? 1 : rows);
cleanup: cleanup:
DBUG_RETURN(ret_val); DBUG_RETURN(ret_val);
} }
......
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