mirror of
https://github.com/MariaDB/server.git
synced 2025-01-18 21:12:26 +01:00
1081513a12
Merge 3.23.52 innobase/btr/btr0btr.c: Merge 3.23.52 innobase/btr/btr0cur.c: Merge 3.23.52 innobase/btr/btr0sea.c: Merge 3.23.52 innobase/include/btr0btr.h: Merge 3.23.52 innobase/include/btr0cur.h: Merge 3.23.52 innobase/include/btr0sea.h: Merge 3.23.52 innobase/include/buf0buf.h: Merge 3.23.52 innobase/include/buf0rea.h: Merge 3.23.52 innobase/include/data0data.h: Merge 3.23.52 innobase/include/data0data.ic: Merge 3.23.52 innobase/include/log0log.h: Merge 3.23.52 innobase/include/log0log.ic: Merge 3.23.52 innobase/include/os0file.h: Merge 3.23.52 innobase/include/page0page.h: Merge 3.23.52 innobase/include/page0page.ic: Merge 3.23.52 innobase/include/row0mysql.h: Merge 3.23.52 innobase/include/trx0roll.h: Merge 3.23.52 innobase/include/trx0sys.h: Merge 3.23.52 innobase/include/trx0trx.h: Merge 3.23.52 innobase/include/ut0ut.h: Merge 3.23.52 innobase/include/univ.i: Merge 3.23.52 innobase/include/ut0ut.ic: Merge 3.23.52 innobase/buf/buf0buf.c: Merge 3.23.52 innobase/buf/buf0rea.c: Merge 3.23.52 innobase/data/data0data.c: Merge 3.23.52 innobase/dict/dict0crea.c: Merge 3.23.52 innobase/dict/dict0dict.c: Merge 3.23.52 innobase/dict/dict0load.c: Merge 3.23.52 innobase/dict/dict0mem.c: Merge 3.23.52 innobase/fsp/fsp0fsp.c: Merge 3.23.52 innobase/ibuf/ibuf0ibuf.c: Merge 3.23.52 innobase/lock/lock0lock.c: Merge 3.23.52 innobase/log/log0log.c: Merge 3.23.52 innobase/log/log0recv.c: Merge 3.23.52 innobase/mtr/mtr0log.c: Merge 3.23.52 innobase/mtr/mtr0mtr.c: Merge 3.23.52 innobase/os/os0file.c: Merge 3.23.52 innobase/page/page0cur.c: Merge 3.23.52 innobase/page/page0page.c: Merge 3.23.52 innobase/rem/rem0cmp.c: Merge 3.23.52 innobase/row/row0ins.c: Merge 3.23.52 innobase/row/row0mysql.c: Merge 3.23.52 innobase/row/row0purge.c: Merge 3.23.52 innobase/row/row0upd.c: Merge 3.23.52 innobase/srv/srv0srv.c: Merge 3.23.52 innobase/srv/srv0start.c: Merge 3.23.52 innobase/trx/trx0roll.c: Merge 3.23.52 innobase/trx/trx0sys.c: Merge 3.23.52 innobase/trx/trx0trx.c: Merge 3.23.52 innobase/trx/trx0undo.c: Merge 3.23.52 innobase/ut/ut0mem.c: Merge 3.23.52 innobase/ut/ut0ut.c: Merge 3.23.52
1134 lines
25 KiB
C
1134 lines
25 KiB
C
/************************************************************************
|
|
SQL data field and tuple
|
|
|
|
(c) 1994-1996 Innobase Oy
|
|
|
|
Created 5/30/1994 Heikki Tuuri
|
|
*************************************************************************/
|
|
|
|
#include "data0data.h"
|
|
|
|
#ifdef UNIV_NONINL
|
|
#include "data0data.ic"
|
|
#endif
|
|
|
|
#include "ut0rnd.h"
|
|
#include "rem0rec.h"
|
|
#include "rem0cmp.h"
|
|
#include "page0page.h"
|
|
#include "dict0dict.h"
|
|
#include "btr0cur.h"
|
|
|
|
byte data_error; /* data pointers of tuple fields are initialized
|
|
to point here for error checking */
|
|
|
|
ulint data_dummy; /* this is used to fool the compiler in
|
|
dtuple_validate */
|
|
|
|
byte data_buf[8192]; /* used in generating test tuples */
|
|
ulint data_rnd = 756511;
|
|
|
|
|
|
/* Some non-inlined functions used in the MySQL interface: */
|
|
void
|
|
dfield_set_data_noninline(
|
|
dfield_t* field, /* in: field */
|
|
void* data, /* in: data */
|
|
ulint len) /* in: length or UNIV_SQL_NULL */
|
|
{
|
|
dfield_set_data(field, data, len);
|
|
}
|
|
void*
|
|
dfield_get_data_noninline(
|
|
dfield_t* field) /* in: field */
|
|
{
|
|
return(dfield_get_data(field));
|
|
}
|
|
ulint
|
|
dfield_get_len_noninline(
|
|
dfield_t* field) /* in: field */
|
|
{
|
|
return(dfield_get_len(field));
|
|
}
|
|
ulint
|
|
dtuple_get_n_fields_noninline(
|
|
dtuple_t* tuple) /* in: tuple */
|
|
{
|
|
return(dtuple_get_n_fields(tuple));
|
|
}
|
|
dfield_t*
|
|
dtuple_get_nth_field_noninline(
|
|
dtuple_t* tuple, /* in: tuple */
|
|
ulint n) /* in: index of field */
|
|
{
|
|
return(dtuple_get_nth_field(tuple, n));
|
|
}
|
|
|
|
/*************************************************************************
|
|
Tests if dfield data length and content is equal to the given. */
|
|
|
|
ibool
|
|
dfield_data_is_binary_equal(
|
|
/*========================*/
|
|
/* out: TRUE if equal */
|
|
dfield_t* field, /* in: field */
|
|
ulint len, /* in: data length or UNIV_SQL_NULL */
|
|
byte* data) /* in: data */
|
|
{
|
|
if (len != field->len) {
|
|
|
|
return(FALSE);
|
|
}
|
|
|
|
if (len == UNIV_SQL_NULL) {
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
if (0 != ut_memcmp(field->data, data, len)) {
|
|
|
|
return(FALSE);
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/****************************************************************
|
|
Returns TRUE if lengths of two dtuples are equal and respective data fields
|
|
in them are equal when compared with collation in char fields (not as binary
|
|
strings). */
|
|
|
|
ibool
|
|
dtuple_datas_are_ordering_equal(
|
|
/*============================*/
|
|
/* out: TRUE if length and fieds are equal
|
|
when compared with cmp_data_data:
|
|
NOTE: in character type fields some letters
|
|
are identified with others! (collation) */
|
|
dtuple_t* tuple1, /* in: tuple 1 */
|
|
dtuple_t* tuple2) /* in: tuple 2 */
|
|
{
|
|
dfield_t* field1;
|
|
dfield_t* field2;
|
|
ulint n_fields;
|
|
ulint i;
|
|
|
|
ut_ad(tuple1 && tuple2);
|
|
ut_ad(tuple1->magic_n = DATA_TUPLE_MAGIC_N);
|
|
ut_ad(tuple2->magic_n = DATA_TUPLE_MAGIC_N);
|
|
ut_ad(dtuple_check_typed(tuple1));
|
|
ut_ad(dtuple_check_typed(tuple2));
|
|
|
|
n_fields = dtuple_get_n_fields(tuple1);
|
|
|
|
if (n_fields != dtuple_get_n_fields(tuple2)) {
|
|
|
|
return(FALSE);
|
|
}
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
|
|
field1 = dtuple_get_nth_field(tuple1, i);
|
|
field2 = dtuple_get_nth_field(tuple2, i);
|
|
|
|
if (0 != cmp_dfield_dfield(field1, field2)) {
|
|
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Creates a dtuple for use in MySQL. */
|
|
|
|
dtuple_t*
|
|
dtuple_create_for_mysql(
|
|
/*====================*/
|
|
/* out, own created dtuple */
|
|
void** heap, /* out: created memory heap */
|
|
ulint n_fields) /* in: number of fields */
|
|
{
|
|
*heap = (void*)mem_heap_create(500);
|
|
|
|
return(dtuple_create(*((mem_heap_t**)heap), n_fields));
|
|
}
|
|
|
|
/*************************************************************************
|
|
Frees a dtuple used in MySQL. */
|
|
|
|
void
|
|
dtuple_free_for_mysql(
|
|
/*==================*/
|
|
void* heap) /* in: memory heap where tuple was created */
|
|
{
|
|
mem_heap_free((mem_heap_t*)heap);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Sets number of fields used in a tuple. Normally this is set in
|
|
dtuple_create, but if you want later to set it smaller, you can use this. */
|
|
|
|
void
|
|
dtuple_set_n_fields(
|
|
/*================*/
|
|
dtuple_t* tuple, /* in: tuple */
|
|
ulint n_fields) /* in: number of fields */
|
|
{
|
|
ut_ad(tuple);
|
|
|
|
tuple->n_fields = n_fields;
|
|
tuple->n_fields_cmp = n_fields;
|
|
}
|
|
|
|
/**************************************************************
|
|
Checks that a data field is typed. */
|
|
static
|
|
ibool
|
|
dfield_check_typed_no_assert(
|
|
/*=========================*/
|
|
/* out: TRUE if ok */
|
|
dfield_t* field) /* in: data field */
|
|
{
|
|
if (dfield_get_type(field)->mtype > DATA_MYSQL
|
|
|| dfield_get_type(field)->mtype < DATA_VARCHAR) {
|
|
|
|
fprintf(stderr,
|
|
"InnoDB: Error: data field type %lu, len %lu\n",
|
|
dfield_get_type(field)->mtype, dfield_get_len(field));
|
|
return(FALSE);
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/**************************************************************
|
|
Checks that a data tuple is typed. */
|
|
|
|
ibool
|
|
dtuple_check_typed_no_assert(
|
|
/*=========================*/
|
|
/* out: TRUE if ok */
|
|
dtuple_t* tuple) /* in: tuple */
|
|
{
|
|
dfield_t* field;
|
|
ulint i;
|
|
char err_buf[1000];
|
|
|
|
if (dtuple_get_n_fields(tuple) > REC_MAX_N_FIELDS) {
|
|
fprintf(stderr,
|
|
"InnoDB: Error: index entry has %lu fields\n",
|
|
dtuple_get_n_fields(tuple));
|
|
|
|
dtuple_sprintf(err_buf, 900, tuple);
|
|
fprintf(stderr,
|
|
"InnoDB: Tuple contents: %s\n", err_buf);
|
|
|
|
return(FALSE);
|
|
}
|
|
|
|
for (i = 0; i < dtuple_get_n_fields(tuple); i++) {
|
|
|
|
field = dtuple_get_nth_field(tuple, i);
|
|
|
|
if (!dfield_check_typed_no_assert(field)) {
|
|
|
|
dtuple_sprintf(err_buf, 900, tuple);
|
|
fprintf(stderr,
|
|
"InnoDB: Tuple contents: %s\n", err_buf);
|
|
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/**************************************************************
|
|
Checks that a data field is typed. Asserts an error if not. */
|
|
|
|
ibool
|
|
dfield_check_typed(
|
|
/*===============*/
|
|
/* out: TRUE if ok */
|
|
dfield_t* field) /* in: data field */
|
|
{
|
|
if (dfield_get_type(field)->mtype > DATA_MYSQL
|
|
|| dfield_get_type(field)->mtype < DATA_VARCHAR) {
|
|
|
|
fprintf(stderr,
|
|
"InnoDB: Error: data field type %lu, len %lu\n",
|
|
dfield_get_type(field)->mtype, dfield_get_len(field));
|
|
|
|
ut_a(0);
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/**************************************************************
|
|
Checks that a data tuple is typed. Asserts an error if not. */
|
|
|
|
ibool
|
|
dtuple_check_typed(
|
|
/*===============*/
|
|
/* out: TRUE if ok */
|
|
dtuple_t* tuple) /* in: tuple */
|
|
{
|
|
dfield_t* field;
|
|
ulint i;
|
|
|
|
for (i = 0; i < dtuple_get_n_fields(tuple); i++) {
|
|
|
|
field = dtuple_get_nth_field(tuple, i);
|
|
|
|
ut_a(dfield_check_typed(field));
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/**************************************************************
|
|
Validates the consistency of a tuple which must be complete, i.e,
|
|
all fields must have been set. */
|
|
|
|
ibool
|
|
dtuple_validate(
|
|
/*============*/
|
|
/* out: TRUE if ok */
|
|
dtuple_t* tuple) /* in: tuple */
|
|
{
|
|
dfield_t* field;
|
|
byte* data;
|
|
ulint n_fields;
|
|
ulint len;
|
|
ulint i;
|
|
ulint j;
|
|
|
|
ut_a(tuple->magic_n = DATA_TUPLE_MAGIC_N);
|
|
|
|
n_fields = dtuple_get_n_fields(tuple);
|
|
|
|
/* We dereference all the data of each field to test
|
|
for memory traps */
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
|
|
field = dtuple_get_nth_field(tuple, i);
|
|
len = dfield_get_len(field);
|
|
|
|
if (len != UNIV_SQL_NULL) {
|
|
|
|
data = field->data;
|
|
|
|
for (j = 0; j < len; j++) {
|
|
|
|
data_dummy += *data; /* fool the compiler not
|
|
to optimize out this
|
|
code */
|
|
data++;
|
|
}
|
|
}
|
|
}
|
|
|
|
ut_a(dtuple_check_typed(tuple));
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/*****************************************************************
|
|
Pretty prints a dfield value according to its data type. */
|
|
|
|
void
|
|
dfield_print(
|
|
/*=========*/
|
|
dfield_t* dfield) /* in: dfield */
|
|
{
|
|
byte* data;
|
|
ulint len;
|
|
ulint mtype;
|
|
ulint i;
|
|
|
|
len = dfield_get_len(dfield);
|
|
data = dfield_get_data(dfield);
|
|
|
|
if (len == UNIV_SQL_NULL) {
|
|
printf("NULL");
|
|
|
|
return;
|
|
}
|
|
|
|
mtype = dtype_get_mtype(dfield_get_type(dfield));
|
|
|
|
if ((mtype == DATA_CHAR) || (mtype == DATA_VARCHAR)) {
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (isprint((char)(*data))) {
|
|
printf("%c", (char)*data);
|
|
} else {
|
|
printf(" ");
|
|
}
|
|
|
|
data++;
|
|
}
|
|
} else if (mtype == DATA_INT) {
|
|
ut_a(len == 4); /* only works for 32-bit integers */
|
|
printf("%i", (int)mach_read_from_4(data));
|
|
} else {
|
|
ut_error;
|
|
}
|
|
}
|
|
|
|
/*****************************************************************
|
|
Pretty prints a dfield value according to its data type. Also the hex string
|
|
is printed if a string contains non-printable characters. */
|
|
|
|
void
|
|
dfield_print_also_hex(
|
|
/*==================*/
|
|
dfield_t* dfield) /* in: dfield */
|
|
{
|
|
byte* data;
|
|
ulint len;
|
|
ulint mtype;
|
|
ulint i;
|
|
ibool print_also_hex;
|
|
|
|
len = dfield_get_len(dfield);
|
|
data = dfield_get_data(dfield);
|
|
|
|
if (len == UNIV_SQL_NULL) {
|
|
printf("NULL");
|
|
|
|
return;
|
|
}
|
|
|
|
mtype = dtype_get_mtype(dfield_get_type(dfield));
|
|
|
|
if ((mtype == DATA_CHAR) || (mtype == DATA_VARCHAR)) {
|
|
|
|
print_also_hex = FALSE;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (isprint((char)(*data))) {
|
|
printf("%c", (char)*data);
|
|
} else {
|
|
print_also_hex = TRUE;
|
|
printf(" ");
|
|
}
|
|
|
|
data++;
|
|
}
|
|
|
|
if (!print_also_hex) {
|
|
|
|
return;
|
|
}
|
|
|
|
printf(" Hex: ");
|
|
|
|
data = dfield_get_data(dfield);
|
|
|
|
for (i = 0; i < len; i++) {
|
|
printf("%02lx", (ulint)*data);
|
|
|
|
data++;
|
|
}
|
|
} else if (mtype == DATA_INT) {
|
|
ut_a(len == 4); /* inly works for 32-bit integers */
|
|
printf("%i", (int)mach_read_from_4(data));
|
|
} else {
|
|
ut_error;
|
|
}
|
|
}
|
|
|
|
/**************************************************************
|
|
The following function prints the contents of a tuple. */
|
|
|
|
void
|
|
dtuple_print(
|
|
/*=========*/
|
|
dtuple_t* tuple) /* in: tuple */
|
|
{
|
|
dfield_t* field;
|
|
ulint n_fields;
|
|
ulint i;
|
|
|
|
n_fields = dtuple_get_n_fields(tuple);
|
|
|
|
printf("DATA TUPLE: %lu fields;\n", n_fields);
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
printf(" %lu:", i);
|
|
|
|
field = dtuple_get_nth_field(tuple, i);
|
|
|
|
if (field->len != UNIV_SQL_NULL) {
|
|
ut_print_buf(field->data, field->len);
|
|
} else {
|
|
printf(" SQL NULL");
|
|
}
|
|
|
|
printf(";");
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
dtuple_validate(tuple);
|
|
}
|
|
|
|
/**************************************************************
|
|
The following function prints the contents of a tuple to a buffer. */
|
|
|
|
ulint
|
|
dtuple_sprintf(
|
|
/*===========*/
|
|
/* out: printed length in bytes */
|
|
char* buf, /* in: print buffer */
|
|
ulint buf_len,/* in: buf length in bytes */
|
|
dtuple_t* tuple) /* in: tuple */
|
|
{
|
|
dfield_t* field;
|
|
ulint n_fields;
|
|
ulint len;
|
|
ulint i;
|
|
|
|
len = 0;
|
|
|
|
n_fields = dtuple_get_n_fields(tuple);
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
if (len + 30 > buf_len) {
|
|
|
|
return(len);
|
|
}
|
|
|
|
len += sprintf(buf + len, " %lu:", i);
|
|
|
|
field = dtuple_get_nth_field(tuple, i);
|
|
|
|
if (field->len != UNIV_SQL_NULL) {
|
|
if (5 * field->len + len + 30 > buf_len) {
|
|
|
|
return(len);
|
|
}
|
|
|
|
len += ut_sprintf_buf(buf + len, field->data,
|
|
field->len);
|
|
} else {
|
|
len += sprintf(buf + len, " SQL NULL");
|
|
}
|
|
|
|
len += sprintf(buf + len, ";");
|
|
}
|
|
|
|
return(len);
|
|
}
|
|
|
|
/******************************************************************
|
|
Moves parts of long fields in entry to the big record vector so that
|
|
the size of tuple drops below the maximum record size allowed in the
|
|
database. Moves data only from those fields which are not necessary
|
|
to determine uniquely the insertion place of the tuple in the index. */
|
|
|
|
big_rec_t*
|
|
dtuple_convert_big_rec(
|
|
/*===================*/
|
|
/* out, own: created big record vector,
|
|
NULL if we are not able to shorten
|
|
the entry enough, i.e., if there are
|
|
too many short fields in entry */
|
|
dict_index_t* index, /* in: index */
|
|
dtuple_t* entry, /* in: index entry */
|
|
ulint* ext_vec,/* in: array of externally stored fields,
|
|
or NULL: if a field already is externally
|
|
stored, then we cannot move it to the vector
|
|
this function returns */
|
|
ulint n_ext_vec)/* in: number of elements is ext_vec */
|
|
{
|
|
mem_heap_t* heap;
|
|
big_rec_t* vector;
|
|
dfield_t* dfield;
|
|
ulint size;
|
|
ulint n_fields;
|
|
ulint longest;
|
|
ulint longest_i = ULINT_MAX;
|
|
ibool is_externally_stored;
|
|
ulint i;
|
|
ulint j;
|
|
char err_buf[1000];
|
|
|
|
ut_a(dtuple_check_typed_no_assert(entry));
|
|
|
|
size = rec_get_converted_size(entry);
|
|
|
|
if (size > 1000000000) {
|
|
fprintf(stderr,
|
|
"InnoDB: Warning: tuple size very big: %lu\n", size);
|
|
|
|
dtuple_sprintf(err_buf, 900, entry);
|
|
fprintf(stderr,
|
|
"InnoDB: Tuple contents: %s\n", err_buf);
|
|
}
|
|
|
|
heap = mem_heap_create(size + dtuple_get_n_fields(entry)
|
|
* sizeof(big_rec_field_t) + 1000);
|
|
|
|
vector = mem_heap_alloc(heap, sizeof(big_rec_t));
|
|
|
|
vector->heap = heap;
|
|
vector->fields = mem_heap_alloc(heap, dtuple_get_n_fields(entry)
|
|
* sizeof(big_rec_field_t));
|
|
|
|
/* Decide which fields to shorten: the algorithm is to look for
|
|
the longest field which does not occur in the ordering part
|
|
of any index on the table */
|
|
|
|
n_fields = 0;
|
|
|
|
while ((rec_get_converted_size(entry)
|
|
>= page_get_free_space_of_empty() / 2)
|
|
|| rec_get_converted_size(entry) >= REC_MAX_DATA_SIZE) {
|
|
|
|
longest = 0;
|
|
for (i = dict_index_get_n_unique_in_tree(index);
|
|
i < dtuple_get_n_fields(entry); i++) {
|
|
|
|
/* Skip over fields which already are externally
|
|
stored */
|
|
|
|
is_externally_stored = FALSE;
|
|
|
|
if (ext_vec) {
|
|
for (j = 0; j < n_ext_vec; j++) {
|
|
if (ext_vec[j] == i) {
|
|
is_externally_stored = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Skip over fields which are ordering in some index */
|
|
|
|
if (!is_externally_stored &&
|
|
dict_field_get_col(
|
|
dict_index_get_nth_field(index, i))
|
|
->ord_part == 0) {
|
|
|
|
dfield = dtuple_get_nth_field(entry, i);
|
|
|
|
if (dfield->len != UNIV_SQL_NULL &&
|
|
dfield->len > longest) {
|
|
|
|
longest = dfield->len;
|
|
|
|
longest_i = i;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (longest < BTR_EXTERN_FIELD_REF_SIZE + 10
|
|
+ REC_1BYTE_OFFS_LIMIT) {
|
|
|
|
/* Cannot shorten more */
|
|
|
|
mem_heap_free(heap);
|
|
|
|
return(NULL);
|
|
}
|
|
|
|
/* Move data from field longest_i to big rec vector;
|
|
we do not let data size of the remaining entry
|
|
drop below 128 which is the limit for the 2-byte
|
|
offset storage format in a physical record. This
|
|
we accomplish by storing 128 bytes of data in entry
|
|
itself, and only the remaining part to big rec vec. */
|
|
|
|
dfield = dtuple_get_nth_field(entry, longest_i);
|
|
vector->fields[n_fields].field_no = longest_i;
|
|
|
|
vector->fields[n_fields].len = dfield->len
|
|
- REC_1BYTE_OFFS_LIMIT;
|
|
|
|
vector->fields[n_fields].data = mem_heap_alloc(heap,
|
|
vector->fields[n_fields].len);
|
|
|
|
/* Copy data (from the end of field) to big rec vector */
|
|
|
|
ut_memcpy(vector->fields[n_fields].data,
|
|
((byte*)dfield->data) + dfield->len
|
|
- vector->fields[n_fields].len,
|
|
vector->fields[n_fields].len);
|
|
dfield->len = dfield->len - vector->fields[n_fields].len
|
|
+ BTR_EXTERN_FIELD_REF_SIZE;
|
|
|
|
/* Set the extern field reference in dfield to zero */
|
|
memset(((byte*)dfield->data)
|
|
+ dfield->len - BTR_EXTERN_FIELD_REF_SIZE,
|
|
0, BTR_EXTERN_FIELD_REF_SIZE);
|
|
n_fields++;
|
|
}
|
|
|
|
vector->n_fields = n_fields;
|
|
return(vector);
|
|
}
|
|
|
|
/******************************************************************
|
|
Puts back to entry the data stored in vector. Note that to ensure the
|
|
fields in entry can accommodate the data, vector must have been created
|
|
from entry with dtuple_convert_big_rec. */
|
|
|
|
void
|
|
dtuple_convert_back_big_rec(
|
|
/*========================*/
|
|
dict_index_t* index, /* in: index */
|
|
dtuple_t* entry, /* in: entry whose data was put to vector */
|
|
big_rec_t* vector) /* in, own: big rec vector; it is
|
|
freed in this function */
|
|
{
|
|
dfield_t* dfield;
|
|
ulint i;
|
|
|
|
for (i = 0; i < vector->n_fields; i++) {
|
|
|
|
dfield = dtuple_get_nth_field(entry,
|
|
vector->fields[i].field_no);
|
|
/* Copy data from big rec vector */
|
|
|
|
ut_memcpy(((byte*)dfield->data)
|
|
+ dfield->len - BTR_EXTERN_FIELD_REF_SIZE,
|
|
vector->fields[i].data,
|
|
vector->fields[i].len);
|
|
dfield->len = dfield->len + vector->fields[i].len
|
|
- BTR_EXTERN_FIELD_REF_SIZE;
|
|
}
|
|
|
|
mem_heap_free(vector->heap);
|
|
}
|
|
|
|
/******************************************************************
|
|
Frees the memory in a big rec vector. */
|
|
|
|
void
|
|
dtuple_big_rec_free(
|
|
/*================*/
|
|
big_rec_t* vector) /* in, own: big rec vector; it is
|
|
freed in this function */
|
|
{
|
|
mem_heap_free(vector->heap);
|
|
}
|
|
|
|
#ifdef notdefined
|
|
|
|
/******************************************************************
|
|
Generates random numbers, where 10/16 is uniformly
|
|
distributed between 0 and n1, 5/16 between 0 and n2,
|
|
and 1/16 between 0 and n3. */
|
|
static
|
|
ulint
|
|
dtuple_gen_rnd_ulint(
|
|
/*=================*/
|
|
/* out: random ulint */
|
|
ulint n1,
|
|
ulint n2,
|
|
ulint n3)
|
|
{
|
|
ulint m;
|
|
ulint n;
|
|
|
|
m = ut_rnd_gen_ulint() % 16;
|
|
|
|
if (m < 10) {
|
|
n = n1;
|
|
} else if (m < 15) {
|
|
n = n2;
|
|
} else {
|
|
n = n3;
|
|
}
|
|
|
|
m = ut_rnd_gen_ulint();
|
|
|
|
return(m % n);
|
|
}
|
|
|
|
/***************************************************************
|
|
Generates a random tuple. */
|
|
|
|
dtuple_t*
|
|
dtuple_gen_rnd_tuple(
|
|
/*=================*/
|
|
/* out: pointer to the tuple */
|
|
mem_heap_t* heap) /* in: memory heap where generated */
|
|
{
|
|
ulint n_fields;
|
|
dfield_t* field;
|
|
ulint len;
|
|
dtuple_t* tuple;
|
|
ulint i;
|
|
ulint j;
|
|
byte* ptr;
|
|
|
|
n_fields = dtuple_gen_rnd_ulint(5, 30, 300) + 1;
|
|
|
|
tuple = dtuple_create(heap, n_fields);
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
|
|
if (n_fields < 7) {
|
|
len = dtuple_gen_rnd_ulint(5, 30, 400);
|
|
} else {
|
|
len = dtuple_gen_rnd_ulint(7, 5, 17);
|
|
}
|
|
|
|
field = dtuple_get_nth_field(tuple, i);
|
|
|
|
if (len == 0) {
|
|
dfield_set_data(field, NULL, UNIV_SQL_NULL);
|
|
} else {
|
|
ptr = mem_heap_alloc(heap, len);
|
|
dfield_set_data(field, ptr, len - 1);
|
|
|
|
for (j = 0; j < len; j++) {
|
|
*ptr = (byte)(65 +
|
|
dtuple_gen_rnd_ulint(22, 22, 22));
|
|
ptr++;
|
|
}
|
|
}
|
|
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR,
|
|
DATA_ENGLISH, 500, 0);
|
|
}
|
|
|
|
ut_a(dtuple_validate(tuple));
|
|
|
|
return(tuple);
|
|
}
|
|
|
|
/*******************************************************************
|
|
Generates a test tuple for sort and comparison tests. */
|
|
|
|
void
|
|
dtuple_gen_test_tuple(
|
|
/*==================*/
|
|
dtuple_t* tuple, /* in/out: a tuple with 3 fields */
|
|
ulint i) /* in: a number < 512 */
|
|
{
|
|
ulint j;
|
|
dfield_t* field;
|
|
void* data = NULL;
|
|
ulint len = 0;
|
|
|
|
for (j = 0; j < 3; j++) {
|
|
switch (i % 8) {
|
|
case 0:
|
|
data = ""; len = 0; break;
|
|
case 1:
|
|
data = "A"; len = 1; break;
|
|
case 2:
|
|
data = "AA"; len = 2; break;
|
|
case 3:
|
|
data = "AB"; len = 2; break;
|
|
case 4:
|
|
data = "B"; len = 1; break;
|
|
case 5:
|
|
data = "BA"; len = 2; break;
|
|
case 6:
|
|
data = "BB"; len = 2; break;
|
|
case 7:
|
|
len = UNIV_SQL_NULL; break;
|
|
}
|
|
|
|
field = dtuple_get_nth_field(tuple, 2 - j);
|
|
|
|
dfield_set_data(field, data, len);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR,
|
|
DATA_ENGLISH, 100, 0);
|
|
|
|
i = i / 8;
|
|
}
|
|
|
|
ut_ad(dtuple_validate(tuple));
|
|
}
|
|
|
|
/*******************************************************************
|
|
Generates a test tuple for B-tree speed tests. */
|
|
|
|
void
|
|
dtuple_gen_test_tuple3(
|
|
/*===================*/
|
|
dtuple_t* tuple, /* in/out: a tuple with >= 3 fields */
|
|
ulint i, /* in: a number < 1000000 */
|
|
ulint type, /* in: DTUPLE_TEST_FIXED30, ... */
|
|
byte* buf) /* in: a buffer of size >= 16 bytes */
|
|
{
|
|
dfield_t* field;
|
|
ulint third_size;
|
|
|
|
ut_ad(tuple && buf);
|
|
ut_ad(i < 1000000);
|
|
|
|
field = dtuple_get_nth_field(tuple, 0);
|
|
|
|
ut_strcpy((char*)buf, "0000000");
|
|
|
|
buf[1] = (byte)('0' + (i / 100000) % 10);
|
|
buf[2] = (byte)('0' + (i / 10000) % 10);
|
|
buf[3] = (byte)('0' + (i / 1000) % 10);
|
|
buf[4] = (byte)('0' + (i / 100) % 10);
|
|
buf[5] = (byte)('0' + (i / 10) % 10);
|
|
buf[6] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf, 8);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
field = dtuple_get_nth_field(tuple, 1);
|
|
|
|
i = i % 1000; /* ut_rnd_gen_ulint() % 1000000; */
|
|
|
|
ut_strcpy((char*)buf + 8, "0000000");
|
|
|
|
buf[9] = (byte)('0' + (i / 100000) % 10);
|
|
buf[10] = (byte)('0' + (i / 10000) % 10);
|
|
buf[11] = (byte)('0' + (i / 1000) % 10);
|
|
buf[12] = (byte)('0' + (i / 100) % 10);
|
|
buf[13] = (byte)('0' + (i / 10) % 10);
|
|
buf[14] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf + 8, 8);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
field = dtuple_get_nth_field(tuple, 2);
|
|
|
|
data_rnd += 8757651;
|
|
|
|
if (type == DTUPLE_TEST_FIXED30) {
|
|
third_size = 30;
|
|
} else if (type == DTUPLE_TEST_RND30) {
|
|
third_size = data_rnd % 30;
|
|
} else if (type == DTUPLE_TEST_RND3500) {
|
|
third_size = data_rnd % 3500;
|
|
} else if (type == DTUPLE_TEST_FIXED2000) {
|
|
third_size = 2000;
|
|
} else if (type == DTUPLE_TEST_FIXED3) {
|
|
third_size = 3;
|
|
} else {
|
|
ut_error;
|
|
}
|
|
|
|
if (type == DTUPLE_TEST_FIXED30) {
|
|
dfield_set_data(field,
|
|
"12345678901234567890123456789", third_size);
|
|
} else {
|
|
dfield_set_data(field, data_buf, third_size);
|
|
}
|
|
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
ut_ad(dtuple_validate(tuple));
|
|
}
|
|
|
|
/*******************************************************************
|
|
Generates a test tuple for B-tree speed tests. */
|
|
|
|
void
|
|
dtuple_gen_search_tuple3(
|
|
/*=====================*/
|
|
dtuple_t* tuple, /* in/out: a tuple with 1 or 2 fields */
|
|
ulint i, /* in: a number < 1000000 */
|
|
byte* buf) /* in: a buffer of size >= 16 bytes */
|
|
{
|
|
dfield_t* field;
|
|
|
|
ut_ad(tuple && buf);
|
|
ut_ad(i < 1000000);
|
|
|
|
field = dtuple_get_nth_field(tuple, 0);
|
|
|
|
ut_strcpy((char*)buf, "0000000");
|
|
|
|
buf[1] = (byte)('0' + (i / 100000) % 10);
|
|
buf[2] = (byte)('0' + (i / 10000) % 10);
|
|
buf[3] = (byte)('0' + (i / 1000) % 10);
|
|
buf[4] = (byte)('0' + (i / 100) % 10);
|
|
buf[5] = (byte)('0' + (i / 10) % 10);
|
|
buf[6] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf, 8);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
if (dtuple_get_n_fields(tuple) == 1) {
|
|
|
|
return;
|
|
}
|
|
|
|
field = dtuple_get_nth_field(tuple, 1);
|
|
|
|
i = (i * 1000) % 1000000;
|
|
|
|
ut_strcpy((char*)buf + 8, "0000000");
|
|
|
|
buf[9] = (byte)('0' + (i / 100000) % 10);
|
|
buf[10] = (byte)('0' + (i / 10000) % 10);
|
|
buf[11] = (byte)('0' + (i / 1000) % 10);
|
|
buf[12] = (byte)('0' + (i / 100) % 10);
|
|
buf[13] = (byte)('0' + (i / 10) % 10);
|
|
buf[14] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf + 8, 8);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
ut_ad(dtuple_validate(tuple));
|
|
}
|
|
|
|
/*******************************************************************
|
|
Generates a test tuple for TPC-A speed test. */
|
|
|
|
void
|
|
dtuple_gen_test_tuple_TPC_A(
|
|
/*========================*/
|
|
dtuple_t* tuple, /* in/out: a tuple with >= 3 fields */
|
|
ulint i, /* in: a number < 10000 */
|
|
byte* buf) /* in: a buffer of size >= 16 bytes */
|
|
{
|
|
dfield_t* field;
|
|
ulint third_size;
|
|
|
|
ut_ad(tuple && buf);
|
|
ut_ad(i < 10000);
|
|
|
|
field = dtuple_get_nth_field(tuple, 0);
|
|
|
|
ut_strcpy((char*)buf, "0000");
|
|
|
|
buf[0] = (byte)('0' + (i / 1000) % 10);
|
|
buf[1] = (byte)('0' + (i / 100) % 10);
|
|
buf[2] = (byte)('0' + (i / 10) % 10);
|
|
buf[3] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf, 5);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
field = dtuple_get_nth_field(tuple, 1);
|
|
|
|
dfield_set_data(field, buf + 8, 5);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
field = dtuple_get_nth_field(tuple, 2);
|
|
|
|
third_size = 90;
|
|
|
|
dfield_set_data(field, data_buf, third_size);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
ut_ad(dtuple_validate(tuple));
|
|
}
|
|
|
|
/*******************************************************************
|
|
Generates a test tuple for B-tree speed tests. */
|
|
|
|
void
|
|
dtuple_gen_search_tuple_TPC_A(
|
|
/*==========================*/
|
|
dtuple_t* tuple, /* in/out: a tuple with 1 field */
|
|
ulint i, /* in: a number < 10000 */
|
|
byte* buf) /* in: a buffer of size >= 16 bytes */
|
|
{
|
|
dfield_t* field;
|
|
|
|
ut_ad(tuple && buf);
|
|
ut_ad(i < 10000);
|
|
|
|
field = dtuple_get_nth_field(tuple, 0);
|
|
|
|
ut_strcpy((char*)buf, "0000");
|
|
|
|
buf[0] = (byte)('0' + (i / 1000) % 10);
|
|
buf[1] = (byte)('0' + (i / 100) % 10);
|
|
buf[2] = (byte)('0' + (i / 10) % 10);
|
|
buf[3] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf, 5);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
ut_ad(dtuple_validate(tuple));
|
|
}
|
|
|
|
/*******************************************************************
|
|
Generates a test tuple for TPC-C speed test. */
|
|
|
|
void
|
|
dtuple_gen_test_tuple_TPC_C(
|
|
/*========================*/
|
|
dtuple_t* tuple, /* in/out: a tuple with >= 12 fields */
|
|
ulint i, /* in: a number < 100000 */
|
|
byte* buf) /* in: a buffer of size >= 16 bytes */
|
|
{
|
|
dfield_t* field;
|
|
ulint size;
|
|
ulint j;
|
|
|
|
ut_ad(tuple && buf);
|
|
ut_ad(i < 100000);
|
|
|
|
field = dtuple_get_nth_field(tuple, 0);
|
|
|
|
buf[0] = (byte)('0' + (i / 10000) % 10);
|
|
buf[1] = (byte)('0' + (i / 1000) % 10);
|
|
buf[2] = (byte)('0' + (i / 100) % 10);
|
|
buf[3] = (byte)('0' + (i / 10) % 10);
|
|
buf[4] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf, 5);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
field = dtuple_get_nth_field(tuple, 1);
|
|
|
|
dfield_set_data(field, buf, 5);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
for (j = 0; j < 10; j++) {
|
|
|
|
field = dtuple_get_nth_field(tuple, 2 + j);
|
|
|
|
size = 24;
|
|
|
|
dfield_set_data(field, data_buf, size);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH,
|
|
100, 0);
|
|
}
|
|
|
|
ut_ad(dtuple_validate(tuple));
|
|
}
|
|
|
|
/*******************************************************************
|
|
Generates a test tuple for B-tree speed tests. */
|
|
|
|
void
|
|
dtuple_gen_search_tuple_TPC_C(
|
|
/*==========================*/
|
|
dtuple_t* tuple, /* in/out: a tuple with 1 field */
|
|
ulint i, /* in: a number < 100000 */
|
|
byte* buf) /* in: a buffer of size >= 16 bytes */
|
|
{
|
|
dfield_t* field;
|
|
|
|
ut_ad(tuple && buf);
|
|
ut_ad(i < 100000);
|
|
|
|
field = dtuple_get_nth_field(tuple, 0);
|
|
|
|
buf[0] = (byte)('0' + (i / 10000) % 10);
|
|
buf[1] = (byte)('0' + (i / 1000) % 10);
|
|
buf[2] = (byte)('0' + (i / 100) % 10);
|
|
buf[3] = (byte)('0' + (i / 10) % 10);
|
|
buf[4] = (byte)('0' + (i % 10));
|
|
|
|
dfield_set_data(field, buf, 5);
|
|
dtype_set(dfield_get_type(field), DATA_VARCHAR, DATA_ENGLISH, 100, 0);
|
|
|
|
ut_ad(dtuple_validate(tuple));
|
|
}
|
|
|
|
#endif /* notdefined */
|