2005-10-27 07:29:40 +00:00
|
|
|
/************************************************************************
|
|
|
|
The page cursor
|
|
|
|
|
|
|
|
(c) 1994-1996 Innobase Oy
|
|
|
|
|
|
|
|
Created 10/4/1994 Heikki Tuuri
|
|
|
|
*************************************************************************/
|
|
|
|
|
|
|
|
#ifndef page0cur_h
|
|
|
|
#define page0cur_h
|
|
|
|
|
|
|
|
#include "univ.i"
|
|
|
|
|
2006-10-20 12:45:53 +00:00
|
|
|
#include "buf0types.h"
|
2005-10-27 07:29:40 +00:00
|
|
|
#include "page0page.h"
|
|
|
|
#include "rem0rec.h"
|
|
|
|
#include "data0data.h"
|
|
|
|
#include "mtr0mtr.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define PAGE_CUR_ADAPT
|
|
|
|
|
|
|
|
/* Page cursor search modes; the values must be in this order! */
|
|
|
|
|
|
|
|
#define PAGE_CUR_G 1
|
|
|
|
#define PAGE_CUR_GE 2
|
|
|
|
#define PAGE_CUR_L 3
|
|
|
|
#define PAGE_CUR_LE 4
|
|
|
|
/*#define PAGE_CUR_LE_OR_EXTENDS 5*/ /* This is a search mode used in
|
|
|
|
"column LIKE 'abc%' ORDER BY column DESC";
|
|
|
|
we have to find strings which are <= 'abc' or
|
|
|
|
which extend it */
|
|
|
|
#ifdef UNIV_SEARCH_DEBUG
|
|
|
|
# define PAGE_CUR_DBG 6 /* As PAGE_CUR_LE, but skips search shortcut */
|
|
|
|
#endif /* UNIV_SEARCH_DEBUG */
|
|
|
|
|
|
|
|
#ifdef PAGE_CUR_ADAPT
|
|
|
|
# ifdef UNIV_SEARCH_PERF_STAT
|
|
|
|
extern ulint page_cur_short_succ;
|
|
|
|
# endif /* UNIV_SEARCH_PERF_STAT */
|
|
|
|
#endif /* PAGE_CUR_ADAPT */
|
|
|
|
|
|
|
|
/*************************************************************
|
|
|
|
Gets pointer to the page frame where the cursor is positioned. */
|
|
|
|
UNIV_INLINE
|
|
|
|
page_t*
|
|
|
|
page_cur_get_page(
|
|
|
|
/*==============*/
|
|
|
|
/* out: page */
|
|
|
|
page_cur_t* cur); /* in: page cursor */
|
|
|
|
/*************************************************************
|
2006-10-20 12:45:53 +00:00
|
|
|
Gets pointer to the buffer block where the cursor is positioned. */
|
|
|
|
UNIV_INLINE
|
|
|
|
buf_block_t*
|
|
|
|
page_cur_get_block(
|
|
|
|
/*===============*/
|
|
|
|
/* out: page */
|
|
|
|
page_cur_t* cur); /* in: page cursor */
|
|
|
|
/*************************************************************
|
|
|
|
Gets pointer to the page frame where the cursor is positioned. */
|
|
|
|
UNIV_INLINE
|
|
|
|
page_zip_des_t*
|
|
|
|
page_cur_get_page_zip(
|
|
|
|
/*==================*/
|
|
|
|
/* out: page */
|
|
|
|
page_cur_t* cur); /* in: page cursor */
|
|
|
|
/*************************************************************
|
2005-10-27 07:29:40 +00:00
|
|
|
Gets the record where the cursor is positioned. */
|
|
|
|
UNIV_INLINE
|
|
|
|
rec_t*
|
|
|
|
page_cur_get_rec(
|
|
|
|
/*=============*/
|
|
|
|
/* out: record */
|
|
|
|
page_cur_t* cur); /* in: page cursor */
|
|
|
|
/*************************************************************
|
2006-02-23 19:25:29 +00:00
|
|
|
Sets the cursor object to point before the first user record
|
2005-10-27 07:29:40 +00:00
|
|
|
on the page. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
page_cur_set_before_first(
|
|
|
|
/*======================*/
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: index page */
|
2005-10-27 07:29:40 +00:00
|
|
|
page_cur_t* cur); /* in: cursor */
|
|
|
|
/*************************************************************
|
2006-02-23 19:25:29 +00:00
|
|
|
Sets the cursor object to point after the last user record on
|
2005-10-27 07:29:40 +00:00
|
|
|
the page. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
page_cur_set_after_last(
|
|
|
|
/*====================*/
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: index page */
|
2005-10-27 07:29:40 +00:00
|
|
|
page_cur_t* cur); /* in: cursor */
|
|
|
|
/*************************************************************
|
|
|
|
Returns TRUE if the cursor is before first user record on page. */
|
|
|
|
UNIV_INLINE
|
|
|
|
ibool
|
|
|
|
page_cur_is_before_first(
|
|
|
|
/*=====================*/
|
|
|
|
/* out: TRUE if at start */
|
|
|
|
const page_cur_t* cur); /* in: cursor */
|
|
|
|
/*************************************************************
|
|
|
|
Returns TRUE if the cursor is after last user record. */
|
|
|
|
UNIV_INLINE
|
|
|
|
ibool
|
|
|
|
page_cur_is_after_last(
|
|
|
|
/*===================*/
|
|
|
|
/* out: TRUE if at end */
|
|
|
|
const page_cur_t* cur); /* in: cursor */
|
|
|
|
/**************************************************************
|
|
|
|
Positions the cursor on the given record. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
page_cur_position(
|
|
|
|
/*==============*/
|
|
|
|
rec_t* rec, /* in: record on a page */
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: buffer block containing the record */
|
|
|
|
page_cur_t* cur); /* out: page cursor */
|
2005-10-27 07:29:40 +00:00
|
|
|
/**************************************************************
|
|
|
|
Invalidates a page cursor by setting the record pointer NULL. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
page_cur_invalidate(
|
|
|
|
/*================*/
|
|
|
|
page_cur_t* cur); /* in: page cursor */
|
|
|
|
/**************************************************************
|
|
|
|
Moves the cursor to the next record on page. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
page_cur_move_to_next(
|
|
|
|
/*==================*/
|
|
|
|
page_cur_t* cur); /* in: cursor; must not be after last */
|
|
|
|
/**************************************************************
|
|
|
|
Moves the cursor to the previous record on page. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
page_cur_move_to_prev(
|
|
|
|
/*==================*/
|
|
|
|
page_cur_t* cur); /* in: cursor; must not before first */
|
|
|
|
/***************************************************************
|
|
|
|
Inserts a record next to page cursor. Returns pointer to inserted record if
|
|
|
|
succeed, i.e., enough space available, NULL otherwise. The cursor stays at
|
2007-05-15 09:31:41 +00:00
|
|
|
the same logical position, but the physical position may change if it is
|
|
|
|
pointing to a compressed page that was reorganized. */
|
2005-10-27 07:29:40 +00:00
|
|
|
UNIV_INLINE
|
|
|
|
rec_t*
|
|
|
|
page_cur_tuple_insert(
|
|
|
|
/*==================*/
|
|
|
|
/* out: pointer to record if succeed, NULL
|
|
|
|
otherwise */
|
2007-05-15 09:31:41 +00:00
|
|
|
page_cur_t* cursor, /* in/out: a page cursor */
|
2006-10-20 08:30:07 +00:00
|
|
|
const dtuple_t* tuple, /* in: pointer to a data tuple */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
branches/zip: Make merge sort handle externally stored columns.
Some things still fail in innodb-index.test, and there seems to be
a race condition (data dictionary lock wait) when running with --valgrind.
dfield_t: Add an "external storage" flag, dfield->ext.
dfield_is_null(), dfield_is_ext(), dfield_set_ext(), dfield_set_null():
New functions.
dfield_copy(), dfield_copy_data(): Add const qualifiers, fix in/out comments.
data_write_sql_null(): Use memset().
big_rec_field_t: Replace byte* data with const void* data.
ut_ulint_sort(): Remove.
upd_field_t: Remove extern_storage.
upd_node_t: Replace ext_vec, n_ext_vec with n_ext.
row_merge_copy_blobs(): New function.
row_ins_index_entry(): Add the parameter "ibool foreign" for suppressing
foreign key checks during fast index creation or when inserting into
secondary indexes.
btr_page_insert_fits(): Add const qualifiers.
btr_cur_add_ext(), upd_ext_vec_contains(): Remove.
dfield_print_also_hex(), dfield_print(): Replace if...else if with switch.
Observe dfield_is_ext().
2007-06-21 09:43:15 +00:00
|
|
|
ulint n_ext, /* in: number of externally stored columns */
|
2006-08-17 11:57:51 +00:00
|
|
|
mtr_t* mtr); /* in: mini-transaction handle, or NULL */
|
2005-10-27 07:29:40 +00:00
|
|
|
/***************************************************************
|
|
|
|
Inserts a record next to page cursor. Returns pointer to inserted record if
|
|
|
|
succeed, i.e., enough space available, NULL otherwise. The cursor stays at
|
2007-05-15 09:31:41 +00:00
|
|
|
the same logical position, but the physical position may change if it is
|
|
|
|
pointing to a compressed page that was reorganized. */
|
2005-10-27 07:29:40 +00:00
|
|
|
UNIV_INLINE
|
|
|
|
rec_t*
|
|
|
|
page_cur_rec_insert(
|
|
|
|
/*================*/
|
|
|
|
/* out: pointer to record if succeed, NULL
|
|
|
|
otherwise */
|
2007-05-15 09:31:41 +00:00
|
|
|
page_cur_t* cursor, /* in/out: a page cursor */
|
|
|
|
const rec_t* rec, /* in: record to insert */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
2006-03-14 14:38:45 +00:00
|
|
|
ulint* offsets,/* in/out: rec_get_offsets(rec, index) */
|
2006-08-17 11:57:51 +00:00
|
|
|
mtr_t* mtr); /* in: mini-transaction handle, or NULL */
|
2005-10-27 07:29:40 +00:00
|
|
|
/***************************************************************
|
2007-02-01 09:53:26 +00:00
|
|
|
Inserts a record next to page cursor on an uncompressed page.
|
|
|
|
Returns pointer to inserted record if succeed, i.e., enough
|
|
|
|
space available, NULL otherwise. The cursor stays at the same position. */
|
2005-10-27 07:29:40 +00:00
|
|
|
|
|
|
|
rec_t*
|
|
|
|
page_cur_insert_rec_low(
|
2007-02-01 09:53:26 +00:00
|
|
|
/*====================*/
|
|
|
|
/* out: pointer to record if succeed, NULL
|
|
|
|
otherwise */
|
|
|
|
rec_t* current_rec,/* in: pointer to current record after
|
|
|
|
which the new record is inserted */
|
|
|
|
dict_index_t* index, /* in: record descriptor */
|
2007-05-15 09:31:41 +00:00
|
|
|
const rec_t* rec, /* in: pointer to a physical record */
|
2007-02-01 09:53:26 +00:00
|
|
|
ulint* offsets,/* in/out: rec_get_offsets(rec, index) */
|
|
|
|
mtr_t* mtr); /* in: mini-transaction handle, or NULL */
|
|
|
|
/***************************************************************
|
|
|
|
Inserts a record next to page cursor on a compressed and uncompressed
|
|
|
|
page. Returns pointer to inserted record if succeed, i.e.,
|
|
|
|
enough space available, NULL otherwise.
|
|
|
|
The cursor stays at the same position. */
|
|
|
|
|
|
|
|
rec_t*
|
|
|
|
page_cur_insert_rec_zip(
|
2005-10-27 07:29:40 +00:00
|
|
|
/*====================*/
|
|
|
|
/* out: pointer to record if succeed, NULL
|
|
|
|
otherwise */
|
2006-10-25 08:45:25 +00:00
|
|
|
rec_t** current_rec,/* in/out: pointer to current record after
|
|
|
|
which the new record is inserted */
|
2007-02-01 09:53:26 +00:00
|
|
|
buf_block_t* block, /* in: buffer block of *current_rec */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
2007-05-15 09:31:41 +00:00
|
|
|
const rec_t* rec, /* in: pointer to a physical record */
|
2006-03-14 14:38:45 +00:00
|
|
|
ulint* offsets,/* in/out: rec_get_offsets(rec, index) */
|
2006-08-17 11:57:51 +00:00
|
|
|
mtr_t* mtr); /* in: mini-transaction handle, or NULL */
|
2005-10-27 07:29:40 +00:00
|
|
|
/*****************************************************************
|
|
|
|
Copies records from page to a newly created page, from a given record onward,
|
|
|
|
including that record. Infimum and supremum records are not copied. */
|
|
|
|
|
|
|
|
void
|
|
|
|
page_copy_rec_list_end_to_created_page(
|
|
|
|
/*===================================*/
|
2006-04-11 12:27:06 +00:00
|
|
|
page_t* new_page, /* in/out: index page to copy to */
|
2005-10-27 07:29:40 +00:00
|
|
|
rec_t* rec, /* in: first record to copy */
|
|
|
|
dict_index_t* index, /* in: record descriptor */
|
|
|
|
mtr_t* mtr); /* in: mtr */
|
|
|
|
/***************************************************************
|
2006-02-23 19:25:29 +00:00
|
|
|
Deletes a record at the page cursor. The cursor is moved to the
|
2005-10-27 07:29:40 +00:00
|
|
|
next record after the deleted one. */
|
|
|
|
|
|
|
|
void
|
|
|
|
page_cur_delete_rec(
|
|
|
|
/*================*/
|
2006-02-23 19:25:29 +00:00
|
|
|
page_cur_t* cursor, /* in/out: a page cursor */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
|
|
|
const ulint* offsets,/* in: rec_get_offsets(cursor->rec, index) */
|
|
|
|
mtr_t* mtr); /* in: mini-transaction handle */
|
|
|
|
/********************************************************************
|
|
|
|
Searches the right position for a page cursor. */
|
|
|
|
UNIV_INLINE
|
|
|
|
ulint
|
|
|
|
page_cur_search(
|
|
|
|
/*============*/
|
|
|
|
/* out: number of matched fields on the left */
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: buffer block */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
2006-10-20 08:30:07 +00:00
|
|
|
const dtuple_t* tuple, /* in: data tuple */
|
2005-10-27 07:29:40 +00:00
|
|
|
ulint mode, /* in: PAGE_CUR_L, PAGE_CUR_LE, PAGE_CUR_G,
|
|
|
|
or PAGE_CUR_GE */
|
|
|
|
page_cur_t* cursor);/* out: page cursor */
|
|
|
|
/********************************************************************
|
|
|
|
Searches the right position for a page cursor. */
|
|
|
|
|
|
|
|
void
|
|
|
|
page_cur_search_with_match(
|
|
|
|
/*=======================*/
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: buffer block */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
2006-10-20 08:30:07 +00:00
|
|
|
const dtuple_t* tuple, /* in: data tuple */
|
2005-10-27 07:29:40 +00:00
|
|
|
ulint mode, /* in: PAGE_CUR_L, PAGE_CUR_LE, PAGE_CUR_G,
|
|
|
|
or PAGE_CUR_GE */
|
|
|
|
ulint* iup_matched_fields,
|
|
|
|
/* in/out: already matched fields in upper
|
|
|
|
limit record */
|
|
|
|
ulint* iup_matched_bytes,
|
|
|
|
/* in/out: already matched bytes in a field
|
|
|
|
not yet completely matched */
|
|
|
|
ulint* ilow_matched_fields,
|
|
|
|
/* in/out: already matched fields in lower
|
|
|
|
limit record */
|
|
|
|
ulint* ilow_matched_bytes,
|
|
|
|
/* in/out: already matched bytes in a field
|
|
|
|
not yet completely matched */
|
2006-02-23 19:25:29 +00:00
|
|
|
page_cur_t* cursor); /* out: page cursor */
|
2005-10-27 07:29:40 +00:00
|
|
|
/***************************************************************
|
|
|
|
Positions a page cursor on a randomly chosen user record on a page. If there
|
|
|
|
are no user records, sets the cursor on the infimum record. */
|
|
|
|
|
|
|
|
void
|
|
|
|
page_cur_open_on_rnd_user_rec(
|
|
|
|
/*==========================*/
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: page */
|
|
|
|
page_cur_t* cursor);/* out: page cursor */
|
2005-10-27 07:29:40 +00:00
|
|
|
/***************************************************************
|
|
|
|
Parses a log record of a record insert on a page. */
|
|
|
|
|
|
|
|
byte*
|
|
|
|
page_cur_parse_insert_rec(
|
|
|
|
/*======================*/
|
|
|
|
/* out: end of log record or NULL */
|
|
|
|
ibool is_short,/* in: TRUE if short inserts */
|
|
|
|
byte* ptr, /* in: buffer */
|
|
|
|
byte* end_ptr,/* in: buffer end */
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: page or NULL */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
|
|
|
mtr_t* mtr); /* in: mtr or NULL */
|
|
|
|
/**************************************************************
|
|
|
|
Parses a log record of copying a record list end to a new created page. */
|
|
|
|
|
|
|
|
byte*
|
|
|
|
page_parse_copy_rec_list_to_created_page(
|
|
|
|
/*=====================================*/
|
|
|
|
/* out: end of log record or NULL */
|
|
|
|
byte* ptr, /* in: buffer */
|
|
|
|
byte* end_ptr,/* in: buffer end */
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: page or NULL */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
|
|
|
mtr_t* mtr); /* in: mtr or NULL */
|
|
|
|
/***************************************************************
|
|
|
|
Parses log record of a record delete on a page. */
|
|
|
|
|
|
|
|
byte*
|
|
|
|
page_cur_parse_delete_rec(
|
|
|
|
/*======================*/
|
|
|
|
/* out: pointer to record end or NULL */
|
|
|
|
byte* ptr, /* in: buffer */
|
|
|
|
byte* end_ptr,/* in: buffer end */
|
2006-10-20 12:45:53 +00:00
|
|
|
buf_block_t* block, /* in: page or NULL */
|
2005-10-27 07:29:40 +00:00
|
|
|
dict_index_t* index, /* in: record descriptor */
|
|
|
|
mtr_t* mtr); /* in: mtr or NULL */
|
|
|
|
|
|
|
|
/* Index page cursor */
|
|
|
|
|
|
|
|
struct page_cur_struct{
|
2006-10-20 12:45:53 +00:00
|
|
|
byte* rec; /* pointer to a record on page */
|
|
|
|
buf_block_t* block; /* pointer to the block containing rec */
|
2005-10-27 07:29:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifndef UNIV_NONINL
|
|
|
|
#include "page0cur.ic"
|
|
|
|
#endif
|
|
|
|
|
2006-02-23 19:25:29 +00:00
|
|
|
#endif
|