mirror of
				https://github.com/MariaDB/server.git
				synced 2025-11-04 04:46:15 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1276 lines
		
	
	
	
		
			42 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1276 lines
		
	
	
	
		
			42 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*****************************************************************************
 | 
						|
 | 
						|
Copyright (c) 1994, 2016, Oracle and/or its affiliates. All Rights Reserved.
 | 
						|
Copyright (c) 2017, 2022, MariaDB Corporation.
 | 
						|
 | 
						|
This program is free software; you can redistribute it and/or modify it under
 | 
						|
the terms of the GNU General Public License as published by the Free Software
 | 
						|
Foundation; version 2 of the License.
 | 
						|
 | 
						|
This program is distributed in the hope that it will be useful, but WITHOUT
 | 
						|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
 | 
						|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
 | 
						|
 | 
						|
You should have received a copy of the GNU General Public License along with
 | 
						|
this program; if not, write to the Free Software Foundation, Inc.,
 | 
						|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA
 | 
						|
 | 
						|
*****************************************************************************/
 | 
						|
 | 
						|
/********************************************************************//**
 | 
						|
@file include/rem0rec.h
 | 
						|
Record manager
 | 
						|
 | 
						|
Created 5/30/1994 Heikki Tuuri
 | 
						|
*************************************************************************/
 | 
						|
 | 
						|
#ifndef rem0rec_h
 | 
						|
#define rem0rec_h
 | 
						|
 | 
						|
#ifndef UNIV_INNOCHECKSUM
 | 
						|
#include "data0data.h"
 | 
						|
#include "rem0types.h"
 | 
						|
#include "mtr0types.h"
 | 
						|
#include "page0types.h"
 | 
						|
#include "dict0dict.h"
 | 
						|
#include "trx0types.h"
 | 
						|
#endif /*! UNIV_INNOCHECKSUM */
 | 
						|
#include <ostream>
 | 
						|
#include <sstream>
 | 
						|
 | 
						|
/* Number of extra bytes in an old-style record,
 | 
						|
in addition to the data and the offsets */
 | 
						|
#define REC_N_OLD_EXTRA_BYTES	6
 | 
						|
/* Number of extra bytes in a new-style record,
 | 
						|
in addition to the data and the offsets */
 | 
						|
#define REC_N_NEW_EXTRA_BYTES	5
 | 
						|
 | 
						|
#define REC_NEW_STATUS		3	/* This is single byte bit-field */
 | 
						|
#define REC_NEW_STATUS_MASK	0x7UL
 | 
						|
#define REC_NEW_STATUS_SHIFT	0
 | 
						|
 | 
						|
/* The following four constants are needed in page0zip.cc in order to
 | 
						|
efficiently compress and decompress pages. */
 | 
						|
 | 
						|
/* The offset of heap_no in a compact record */
 | 
						|
#define REC_NEW_HEAP_NO		4
 | 
						|
/* The shift of heap_no in a compact record.
 | 
						|
The status is stored in the low-order bits. */
 | 
						|
#define	REC_HEAP_NO_SHIFT	3
 | 
						|
 | 
						|
/* Length of a B-tree node pointer, in bytes */
 | 
						|
#define REC_NODE_PTR_SIZE	4
 | 
						|
 | 
						|
#ifndef UNIV_INNOCHECKSUM
 | 
						|
/** SQL null flag in a 1-byte offset of ROW_FORMAT=REDUNDANT records */
 | 
						|
constexpr rec_offs REC_1BYTE_SQL_NULL_MASK= 0x80;
 | 
						|
/** SQL null flag in a 2-byte offset of ROW_FORMAT=REDUNDANT records */
 | 
						|
constexpr rec_offs REC_2BYTE_SQL_NULL_MASK= 0x8000;
 | 
						|
 | 
						|
/** In a 2-byte offset of ROW_FORMAT=REDUNDANT records, the second most
 | 
						|
significant bit denotes that the tail of a field is stored off-page. */
 | 
						|
constexpr rec_offs REC_2BYTE_EXTERN_MASK= 0x4000;
 | 
						|
 | 
						|
constexpr size_t RECORD_OFFSET= 2;
 | 
						|
constexpr size_t INDEX_OFFSET=
 | 
						|
    RECORD_OFFSET + sizeof(rec_t *) / sizeof(rec_offs);
 | 
						|
#endif /* UNIV_INNOCHECKSUM */
 | 
						|
 | 
						|
/* Length of the rec_get_offsets() header */
 | 
						|
constexpr size_t REC_OFFS_HEADER_SIZE=
 | 
						|
#ifdef UNIV_DEBUG
 | 
						|
#ifndef UNIV_INNOCHECKSUM
 | 
						|
    sizeof(rec_t *) / sizeof(rec_offs) +
 | 
						|
    sizeof(dict_index_t *) / sizeof(rec_offs) +
 | 
						|
#endif /* UNIV_INNOCHECKSUM */
 | 
						|
#endif /* UNIV_DEBUG */
 | 
						|
    2;
 | 
						|
 | 
						|
/* Number of elements that should be initially allocated for the
 | 
						|
offsets[] array, first passed to rec_get_offsets() */
 | 
						|
constexpr size_t REC_OFFS_NORMAL_SIZE= 300;
 | 
						|
constexpr size_t REC_OFFS_SMALL_SIZE= 18;
 | 
						|
constexpr size_t REC_OFFS_SEC_INDEX_SIZE=
 | 
						|
    /* PK max key parts */ 16 + /* sec idx max key parts */ 16 +
 | 
						|
    /* child page number for non-leaf pages */ 1;
 | 
						|
 | 
						|
/** Get the base address of offsets.  The extra_size is stored at
 | 
						|
this position, and following positions hold the end offsets of
 | 
						|
the fields. */
 | 
						|
#define rec_offs_base(offsets) (offsets + REC_OFFS_HEADER_SIZE)
 | 
						|
 | 
						|
#ifndef UNIV_INNOCHECKSUM
 | 
						|
/* Offset consists of two parts: 2 upper bits is type and all other bits is
 | 
						|
value */
 | 
						|
 | 
						|
/** Only 4 different values is possible! */
 | 
						|
enum field_type_t
 | 
						|
{
 | 
						|
  /** normal field */
 | 
						|
  STORED_IN_RECORD= 0 << 14,
 | 
						|
  /** this field is stored off-page */
 | 
						|
  STORED_OFFPAGE= 1 << 14,
 | 
						|
  /** just an SQL NULL */
 | 
						|
  SQL_NULL= 2 << 14,
 | 
						|
  /** instantly added field */
 | 
						|
  DEFAULT= 3 << 14,
 | 
						|
};
 | 
						|
 | 
						|
/** without 2 upper bits */
 | 
						|
static constexpr rec_offs DATA_MASK= 0x3fff;
 | 
						|
/** 2 upper bits */
 | 
						|
static constexpr rec_offs TYPE_MASK= ~DATA_MASK;
 | 
						|
inline field_type_t get_type(rec_offs n)
 | 
						|
{
 | 
						|
  return static_cast<field_type_t>(n & TYPE_MASK);
 | 
						|
}
 | 
						|
inline void set_type(rec_offs &n, field_type_t type)
 | 
						|
{
 | 
						|
  n= static_cast<rec_offs>((n & DATA_MASK) | type);
 | 
						|
}
 | 
						|
inline rec_offs get_value(rec_offs n) { return n & DATA_MASK; }
 | 
						|
inline rec_offs combine(rec_offs value, field_type_t type)
 | 
						|
{
 | 
						|
  return static_cast<rec_offs>(get_value(value) | type);
 | 
						|
}
 | 
						|
 | 
						|
/** Compact flag ORed to the extra size returned by rec_get_offsets() */
 | 
						|
constexpr rec_offs REC_OFFS_COMPACT= rec_offs(~(rec_offs(~0) >> 1));
 | 
						|
/** External flag in offsets returned by rec_get_offsets() */
 | 
						|
constexpr rec_offs REC_OFFS_EXTERNAL= REC_OFFS_COMPACT >> 1;
 | 
						|
/** Default value flag in offsets returned by rec_get_offsets() */
 | 
						|
constexpr rec_offs REC_OFFS_DEFAULT= REC_OFFS_COMPACT >> 2;
 | 
						|
constexpr rec_offs REC_OFFS_MASK= REC_OFFS_DEFAULT - 1;
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to get the offset of the
 | 
						|
next chained record on the same page.
 | 
						|
@return the page offset of the next chained record, or 0 if none */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_next_offs(
 | 
						|
/*==============*/
 | 
						|
	const rec_t*	rec,	/*!< in: physical record */
 | 
						|
	ulint		comp)	/*!< in: nonzero=compact page format */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to set the next record offset field
 | 
						|
of an old-style record. */
 | 
						|
UNIV_INLINE
 | 
						|
void
 | 
						|
rec_set_next_offs_old(
 | 
						|
/*==================*/
 | 
						|
	rec_t*	rec,	/*!< in: old-style physical record */
 | 
						|
	ulint	next)	/*!< in: offset of the next record */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to set the next record offset field
 | 
						|
of a new-style record. */
 | 
						|
UNIV_INLINE
 | 
						|
void
 | 
						|
rec_set_next_offs_new(
 | 
						|
/*==================*/
 | 
						|
	rec_t*	rec,	/*!< in/out: new-style physical record */
 | 
						|
	ulint	next)	/*!< in: offset of the next record */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to get the number of fields
 | 
						|
in an old-style record.
 | 
						|
@return number of data fields */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_n_fields_old(
 | 
						|
/*=================*/
 | 
						|
	const rec_t*	rec)	/*!< in: physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to get the number of fields
 | 
						|
in a record.
 | 
						|
@return number of data fields */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_n_fields(
 | 
						|
/*=============*/
 | 
						|
	const rec_t*		rec,	/*!< in: physical record */
 | 
						|
	const dict_index_t*	index)	/*!< in: record descriptor */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/** Confirms the n_fields of the entry is sane with comparing the other
 | 
						|
record in the same page specified
 | 
						|
@param[in]	index	index
 | 
						|
@param[in]	rec	record of the same page
 | 
						|
@param[in]	entry	index entry
 | 
						|
@return	true if n_fields is sane */
 | 
						|
UNIV_INLINE
 | 
						|
bool
 | 
						|
rec_n_fields_is_sane(
 | 
						|
	dict_index_t*	index,
 | 
						|
	const rec_t*	rec,
 | 
						|
	const dtuple_t*	entry)
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to get the number of records owned by the
 | 
						|
previous directory record.
 | 
						|
@return number of owned records */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_n_owned_old(
 | 
						|
/*================*/
 | 
						|
	const rec_t*	rec)	/*!< in: old-style physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to get the number of records owned by the
 | 
						|
previous directory record.
 | 
						|
@return number of owned records */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_n_owned_new(
 | 
						|
/*================*/
 | 
						|
	const rec_t*	rec)	/*!< in: new-style physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to retrieve the info bits of
 | 
						|
a record.
 | 
						|
@return info bits */
 | 
						|
UNIV_INLINE
 | 
						|
byte
 | 
						|
rec_get_info_bits(
 | 
						|
/*==============*/
 | 
						|
	const rec_t*	rec,	/*!< in: physical record */
 | 
						|
	ulint		comp)	/*!< in: nonzero=compact page format */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/** Determine the status bits of a non-REDUNDANT record.
 | 
						|
@param[in]	rec	ROW_FORMAT=COMPACT,DYNAMIC,COMPRESSED record
 | 
						|
@return status bits */
 | 
						|
inline
 | 
						|
rec_comp_status_t
 | 
						|
rec_get_status(const rec_t* rec)
 | 
						|
{
 | 
						|
	byte bits = rec[-REC_NEW_STATUS] & REC_NEW_STATUS_MASK;
 | 
						|
	ut_ad(bits <= REC_STATUS_INSTANT);
 | 
						|
	return static_cast<rec_comp_status_t>(bits);
 | 
						|
}
 | 
						|
 | 
						|
/** Set the status bits of a non-REDUNDANT record.
 | 
						|
@param[in,out]	rec	ROW_FORMAT=COMPACT,DYNAMIC,COMPRESSED record
 | 
						|
@param[in]	bits	status bits */
 | 
						|
inline void rec_set_status(rec_t *rec, byte bits)
 | 
						|
{
 | 
						|
  ut_ad(bits <= REC_STATUS_INSTANT);
 | 
						|
  rec[-REC_NEW_STATUS]= static_cast<byte>((rec[-REC_NEW_STATUS] &
 | 
						|
                                           ~REC_NEW_STATUS_MASK) | bits);
 | 
						|
}
 | 
						|
 | 
						|
/** Get the length of added field count in a REC_STATUS_INSTANT record.
 | 
						|
@param[in]	n_add_field	number of added fields, minus one
 | 
						|
@return	storage size of the field count, in bytes */
 | 
						|
inline unsigned rec_get_n_add_field_len(ulint n_add_field)
 | 
						|
{
 | 
						|
	ut_ad(n_add_field < REC_MAX_N_FIELDS);
 | 
						|
	return n_add_field < 0x80 ? 1 : 2;
 | 
						|
}
 | 
						|
 | 
						|
/** Get the added field count in a REC_STATUS_INSTANT record.
 | 
						|
@param[in,out]	header	variable header of a REC_STATUS_INSTANT record
 | 
						|
@return	number of added fields */
 | 
						|
inline unsigned rec_get_n_add_field(const byte*& header)
 | 
						|
{
 | 
						|
	unsigned n_fields_add = *--header;
 | 
						|
	if (n_fields_add < 0x80) {
 | 
						|
		ut_ad(rec_get_n_add_field_len(n_fields_add) == 1);
 | 
						|
		return n_fields_add;
 | 
						|
	}
 | 
						|
 | 
						|
	n_fields_add &= 0x7f;
 | 
						|
	n_fields_add |= unsigned(*--header) << 7;
 | 
						|
	ut_ad(n_fields_add < REC_MAX_N_FIELDS);
 | 
						|
	ut_ad(rec_get_n_add_field_len(n_fields_add) == 2);
 | 
						|
	return n_fields_add;
 | 
						|
}
 | 
						|
 | 
						|
/** Set the added field count in a REC_STATUS_INSTANT record.
 | 
						|
@param[in,out]	header	variable header of a REC_STATUS_INSTANT record
 | 
						|
@param[in]	n_add	number of added fields, minus 1
 | 
						|
@return	record header before the number of added fields */
 | 
						|
inline void rec_set_n_add_field(byte*& header, ulint n_add)
 | 
						|
{
 | 
						|
	ut_ad(n_add < REC_MAX_N_FIELDS);
 | 
						|
 | 
						|
	if (n_add < 0x80) {
 | 
						|
		*header-- = byte(n_add);
 | 
						|
	} else {
 | 
						|
		*header-- = byte(byte(n_add) | 0x80);
 | 
						|
		*header-- = byte(n_add >> 7);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to retrieve the info and status
 | 
						|
bits of a record.  (Only compact records have status bits.)
 | 
						|
@return info and status bits */
 | 
						|
UNIV_INLINE
 | 
						|
byte
 | 
						|
rec_get_info_and_status_bits(
 | 
						|
/*=========================*/
 | 
						|
	const rec_t*	rec,	/*!< in: physical record */
 | 
						|
	ulint		comp)	/*!< in: nonzero=compact page format */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to set the info and status
 | 
						|
bits of a record.  (Only compact records have status bits.) */
 | 
						|
UNIV_INLINE
 | 
						|
void
 | 
						|
rec_set_info_and_status_bits(
 | 
						|
/*=========================*/
 | 
						|
	rec_t*	rec,	/*!< in/out: compact physical record */
 | 
						|
	ulint	bits)	/*!< in: info bits */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
The following function tells if record is delete marked.
 | 
						|
@return nonzero if delete marked */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_deleted_flag(
 | 
						|
/*=================*/
 | 
						|
	const rec_t*	rec,	/*!< in: physical record */
 | 
						|
	ulint		comp)	/*!< in: nonzero=compact page format */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function tells if a new-style record is a node pointer.
 | 
						|
@return TRUE if node pointer */
 | 
						|
UNIV_INLINE
 | 
						|
bool
 | 
						|
rec_get_node_ptr_flag(
 | 
						|
/*==================*/
 | 
						|
	const rec_t*	rec)	/*!< in: physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to get the order number
 | 
						|
of an old-style record in the heap of the index page.
 | 
						|
@return heap order number */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_heap_no_old(
 | 
						|
/*================*/
 | 
						|
	const rec_t*	rec)	/*!< in: physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to get the order number
 | 
						|
of a new-style record in the heap of the index page.
 | 
						|
@return heap order number */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_heap_no_new(
 | 
						|
/*================*/
 | 
						|
	const rec_t*	rec)	/*!< in: physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to test whether the data offsets
 | 
						|
in the record are stored in one-byte or two-byte format.
 | 
						|
@return TRUE if 1-byte form */
 | 
						|
UNIV_INLINE
 | 
						|
ibool
 | 
						|
rec_get_1byte_offs_flag(
 | 
						|
/*====================*/
 | 
						|
	const rec_t*	rec)	/*!< in: physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
The following function is used to set the 1-byte offsets flag. */
 | 
						|
UNIV_INLINE
 | 
						|
void
 | 
						|
rec_set_1byte_offs_flag(
 | 
						|
/*====================*/
 | 
						|
	rec_t*	rec,	/*!< in: physical record */
 | 
						|
	ibool	flag)	/*!< in: TRUE if 1byte form */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
Returns the offset of nth field end if the record is stored in the 1-byte
 | 
						|
offsets form. If the field is SQL null, the flag is ORed in the returned
 | 
						|
value.
 | 
						|
@return offset of the start of the field, SQL null flag ORed */
 | 
						|
UNIV_INLINE
 | 
						|
uint8_t
 | 
						|
rec_1_get_field_end_info(
 | 
						|
/*=====================*/
 | 
						|
	const rec_t*	rec,	/*!< in: record */
 | 
						|
	ulint		n)	/*!< in: field index */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
Returns the offset of nth field end if the record is stored in the 2-byte
 | 
						|
offsets form. If the field is SQL null, the flag is ORed in the returned
 | 
						|
value.
 | 
						|
@return offset of the start of the field, SQL null flag and extern
 | 
						|
storage flag ORed */
 | 
						|
UNIV_INLINE
 | 
						|
uint16_t
 | 
						|
rec_2_get_field_end_info(
 | 
						|
/*=====================*/
 | 
						|
	const rec_t*	rec,	/*!< in: record */
 | 
						|
	ulint		n)	/*!< in: field index */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
Returns nonzero if the field is stored off-page.
 | 
						|
@retval 0 if the field is stored in-page
 | 
						|
@retval REC_2BYTE_EXTERN_MASK if the field is stored externally */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_2_is_field_extern(
 | 
						|
/*==================*/
 | 
						|
	const rec_t*	rec,	/*!< in: record */
 | 
						|
	ulint		n)	/*!< in: field index */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
Determine how many of the first n columns in a compact
 | 
						|
physical record are stored externally.
 | 
						|
@return number of externally stored columns */
 | 
						|
ulint
 | 
						|
rec_get_n_extern_new(
 | 
						|
/*=================*/
 | 
						|
	const rec_t*		rec,	/*!< in: compact physical record */
 | 
						|
	const dict_index_t*	index,	/*!< in: record descriptor */
 | 
						|
	ulint			n)	/*!< in: number of columns to scan */
 | 
						|
	MY_ATTRIBUTE((nonnull, warn_unused_result));
 | 
						|
 | 
						|
/** Determine the offsets to each field in an index record.
 | 
						|
@param[in]	rec		physical record
 | 
						|
@param[in]	index		the index that the record belongs to
 | 
						|
@param[in,out]	offsets		array comprising offsets[0] allocated elements,
 | 
						|
				or an array from rec_get_offsets(), or NULL
 | 
						|
@param[in]	n_core		0, or index->n_core_fields for leaf page
 | 
						|
@param[in]	n_fields	maximum number of offsets to compute
 | 
						|
				(ULINT_UNDEFINED to compute all offsets)
 | 
						|
@param[in,out]	heap		memory heap
 | 
						|
@return the new offsets */
 | 
						|
rec_offs*
 | 
						|
rec_get_offsets_func(
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	rec_offs*		offsets,
 | 
						|
	ulint			n_core,
 | 
						|
	ulint			n_fields,
 | 
						|
#ifdef UNIV_DEBUG
 | 
						|
	const char*		file,	/*!< in: file name where called */
 | 
						|
	unsigned		line,	/*!< in: line number where called */
 | 
						|
#endif /* UNIV_DEBUG */
 | 
						|
	mem_heap_t**		heap)	/*!< in/out: memory heap */
 | 
						|
#ifdef UNIV_DEBUG
 | 
						|
	MY_ATTRIBUTE((nonnull(1,2,6,8),warn_unused_result));
 | 
						|
#else /* UNIV_DEBUG */
 | 
						|
	MY_ATTRIBUTE((nonnull(1,2,6),warn_unused_result));
 | 
						|
#endif /* UNIV_DEBUG */
 | 
						|
 | 
						|
#ifdef UNIV_DEBUG
 | 
						|
# define rec_get_offsets(rec, index, offsets, leaf, n, heap)		\
 | 
						|
	rec_get_offsets_func(rec,index,offsets,leaf,n,__FILE__,__LINE__,heap)
 | 
						|
#else /* UNIV_DEBUG */
 | 
						|
# define rec_get_offsets(rec, index, offsets, leaf, n, heap)		\
 | 
						|
	rec_get_offsets_func(rec, index, offsets, leaf, n, heap)
 | 
						|
#endif /* UNIV_DEBUG */
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
The following function determines the offsets to each field
 | 
						|
in the record.  It can reuse a previously allocated array. */
 | 
						|
void
 | 
						|
rec_get_offsets_reverse(
 | 
						|
/*====================*/
 | 
						|
	const byte*		extra,	/*!< in: the extra bytes of a
 | 
						|
					compact record in reverse order,
 | 
						|
					excluding the fixed-size
 | 
						|
					REC_N_NEW_EXTRA_BYTES */
 | 
						|
	const dict_index_t*	index,	/*!< in: record descriptor */
 | 
						|
	ulint			node_ptr,/*!< in: nonzero=node pointer,
 | 
						|
					0=leaf node */
 | 
						|
	rec_offs*		offsets)/*!< in/out: array consisting of
 | 
						|
					offsets[0] allocated elements */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
#ifdef UNIV_DEBUG
 | 
						|
/** Validate offsets returned by rec_get_offsets().
 | 
						|
@param[in]	rec	record, or NULL
 | 
						|
@param[in]	index	the index that the record belongs in, or NULL
 | 
						|
@param[in,out]	offsets	the offsets of the record
 | 
						|
@return true */
 | 
						|
bool
 | 
						|
rec_offs_validate(
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	const rec_offs*		offsets)
 | 
						|
	MY_ATTRIBUTE((nonnull(3), warn_unused_result));
 | 
						|
/** Update debug data in offsets, in order to tame rec_offs_validate().
 | 
						|
@param[in]	rec	record
 | 
						|
@param[in]	index	the index that the record belongs in
 | 
						|
@param[in]	leaf	whether the record resides in a leaf page
 | 
						|
@param[in,out]	offsets	offsets from rec_get_offsets() to adjust */
 | 
						|
void
 | 
						|
rec_offs_make_valid(
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	bool			leaf,
 | 
						|
	rec_offs*		offsets)
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
#else
 | 
						|
# define rec_offs_make_valid(rec, index, leaf, offsets)
 | 
						|
#endif /* UNIV_DEBUG */
 | 
						|
 | 
						|
/************************************************************//**
 | 
						|
The following function is used to get the offset to the nth
 | 
						|
data field in an old-style record.
 | 
						|
@return offset to the field */
 | 
						|
ulint
 | 
						|
rec_get_nth_field_offs_old(
 | 
						|
/*=======================*/
 | 
						|
	const rec_t*	rec,	/*!< in: record */
 | 
						|
	ulint		n,	/*!< in: index of the field */
 | 
						|
	ulint*		len)	/*!< out: length of the field; UNIV_SQL_NULL
 | 
						|
				if SQL null */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
#define rec_get_nth_field_old(rec, n, len) \
 | 
						|
((rec) + rec_get_nth_field_offs_old(rec, n, len))
 | 
						|
/************************************************************//**
 | 
						|
Gets the physical size of an old-style field.
 | 
						|
Also an SQL null may have a field of size > 0,
 | 
						|
if the data type is of a fixed size.
 | 
						|
@return field size in bytes */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_nth_field_size(
 | 
						|
/*===================*/
 | 
						|
	const rec_t*	rec,	/*!< in: record */
 | 
						|
	ulint		n)	/*!< in: index of the field */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/************************************************************//**
 | 
						|
The following function is used to get an offset to the nth
 | 
						|
data field in a record.
 | 
						|
@return offset from the origin of rec */
 | 
						|
UNIV_INLINE
 | 
						|
rec_offs
 | 
						|
rec_get_nth_field_offs(
 | 
						|
/*===================*/
 | 
						|
	const rec_offs*	offsets,/*!< in: array returned by rec_get_offsets() */
 | 
						|
	ulint		n,	/*!< in: index of the field */
 | 
						|
	ulint*		len)	/*!< out: length of the field; UNIV_SQL_NULL
 | 
						|
				if SQL null */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
#define rec_get_nth_field(rec, offsets, n, len) \
 | 
						|
((rec) + rec_get_nth_field_offs(offsets, n, len))
 | 
						|
/******************************************************//**
 | 
						|
Determine if the offsets are for a record containing null BLOB pointers.
 | 
						|
@return first field containing a null BLOB pointer, or NULL if none found */
 | 
						|
UNIV_INLINE
 | 
						|
const byte*
 | 
						|
rec_offs_any_null_extern(
 | 
						|
/*=====================*/
 | 
						|
	const rec_t*	rec,		/*!< in: record */
 | 
						|
	const rec_offs*	offsets)	/*!< in: rec_get_offsets(rec) */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/** Mark the nth field as externally stored.
 | 
						|
@param[in]	offsets		array returned by rec_get_offsets()
 | 
						|
@param[in]	n		nth field */
 | 
						|
void
 | 
						|
rec_offs_make_nth_extern(
 | 
						|
        rec_offs*	offsets,
 | 
						|
        const ulint     n);
 | 
						|
 | 
						|
MY_ATTRIBUTE((nonnull))
 | 
						|
/** Determine the number of allocated elements for an array of offsets.
 | 
						|
@param[in]	offsets		offsets after rec_offs_set_n_alloc()
 | 
						|
@return number of elements */
 | 
						|
inline ulint rec_offs_get_n_alloc(const rec_offs *offsets)
 | 
						|
{
 | 
						|
  ut_ad(offsets);
 | 
						|
  ulint n_alloc= offsets[0];
 | 
						|
  ut_ad(n_alloc > REC_OFFS_HEADER_SIZE);
 | 
						|
  MEM_CHECK_ADDRESSABLE(offsets, n_alloc * sizeof *offsets);
 | 
						|
  return n_alloc;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine the number of fields for which offsets have been initialized.
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@return number of fields */
 | 
						|
inline
 | 
						|
ulint
 | 
						|
rec_offs_n_fields(const rec_offs* offsets)
 | 
						|
{
 | 
						|
	ulint	n_fields;
 | 
						|
	ut_ad(offsets);
 | 
						|
	n_fields = offsets[1];
 | 
						|
	ut_ad(n_fields > 0);
 | 
						|
	ut_ad(n_fields <= REC_MAX_N_FIELDS);
 | 
						|
	ut_ad(n_fields + REC_OFFS_HEADER_SIZE
 | 
						|
	      <= rec_offs_get_n_alloc(offsets));
 | 
						|
	return(n_fields);
 | 
						|
}
 | 
						|
 | 
						|
/** Get a flag of a record field.
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@param[in]	n	nth field
 | 
						|
@param[in]	flag	flag to extract
 | 
						|
@return	type of the record field */
 | 
						|
inline field_type_t rec_offs_nth_type(const rec_offs *offsets, ulint n)
 | 
						|
{
 | 
						|
  ut_ad(rec_offs_validate(NULL, NULL, offsets));
 | 
						|
  ut_ad(n < rec_offs_n_fields(offsets));
 | 
						|
  return get_type(rec_offs_base(offsets)[1 + n]);
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if a record field is missing
 | 
						|
(should be replaced by dict_index_t::instant_field_value()).
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@param[in]	n	nth field
 | 
						|
@return	nonzero if default bit is set */
 | 
						|
inline ulint rec_offs_nth_default(const rec_offs *offsets, ulint n)
 | 
						|
{
 | 
						|
  return rec_offs_nth_type(offsets, n) == DEFAULT;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if a record field is SQL NULL
 | 
						|
(should be replaced by dict_index_t::instant_field_value()).
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@param[in]	n	nth field
 | 
						|
@return	nonzero if SQL NULL set */
 | 
						|
inline ulint rec_offs_nth_sql_null(const rec_offs *offsets, ulint n)
 | 
						|
{
 | 
						|
  return rec_offs_nth_type(offsets, n) == SQL_NULL;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if a record field is stored off-page.
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@param[in]	n	nth field
 | 
						|
Returns nonzero if the extern bit is set in nth field of rec.
 | 
						|
@return nonzero if externally stored */
 | 
						|
inline ulint rec_offs_nth_extern(const rec_offs *offsets, ulint n)
 | 
						|
{
 | 
						|
  return rec_offs_nth_type(offsets, n) == STORED_OFFPAGE;
 | 
						|
}
 | 
						|
 | 
						|
/** Get a global flag of a record.
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@param[in]	flag	flag to extract
 | 
						|
@return	the flag of the record field */
 | 
						|
inline ulint rec_offs_any_flag(const rec_offs *offsets, ulint flag)
 | 
						|
{
 | 
						|
  ut_ad(rec_offs_validate(NULL, NULL, offsets));
 | 
						|
  return *rec_offs_base(offsets) & flag;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the offsets are for a record containing off-page columns.
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@return nonzero if any off-page columns exist */
 | 
						|
inline bool rec_offs_any_extern(const rec_offs *offsets)
 | 
						|
{
 | 
						|
  return rec_offs_any_flag(offsets, REC_OFFS_EXTERNAL);
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the offsets are for a record that is missing fields.
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@return nonzero if any fields need to be replaced with
 | 
						|
		dict_index_t::instant_field_value() */
 | 
						|
inline ulint rec_offs_any_default(const rec_offs *offsets)
 | 
						|
{
 | 
						|
  return rec_offs_any_flag(offsets, REC_OFFS_DEFAULT);
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the offsets are for other than ROW_FORMAT=REDUNDANT.
 | 
						|
@param[in]	offsets	rec_get_offsets()
 | 
						|
@return	nonzero	if ROW_FORMAT is COMPACT,DYNAMIC or COMPRESSED
 | 
						|
@retval	0	if ROW_FORMAT=REDUNDANT */
 | 
						|
inline ulint rec_offs_comp(const rec_offs *offsets)
 | 
						|
{
 | 
						|
  ut_ad(rec_offs_validate(NULL, NULL, offsets));
 | 
						|
  return (*rec_offs_base(offsets) & REC_OFFS_COMPACT);
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the record is the metadata pseudo-record
 | 
						|
in the clustered index for instant ADD COLUMN or ALTER TABLE.
 | 
						|
@param[in]	rec	leaf page record
 | 
						|
@param[in]	comp	0 if ROW_FORMAT=REDUNDANT, else nonzero
 | 
						|
@return	whether the record is the metadata pseudo-record */
 | 
						|
inline bool rec_is_metadata(const rec_t* rec, ulint comp)
 | 
						|
{
 | 
						|
	bool is = !!(rec_get_info_bits(rec, comp) & REC_INFO_MIN_REC_FLAG);
 | 
						|
	ut_ad(!is || !comp || rec_get_status(rec) == REC_STATUS_INSTANT);
 | 
						|
	return is;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the record is the metadata pseudo-record
 | 
						|
in the clustered index for instant ADD COLUMN or ALTER TABLE.
 | 
						|
@param[in]	rec	leaf page record
 | 
						|
@param[in]	index	index of the record
 | 
						|
@return	whether the record is the metadata pseudo-record */
 | 
						|
inline bool rec_is_metadata(const rec_t *rec, const dict_index_t &index)
 | 
						|
{
 | 
						|
  return rec_is_metadata(rec, index.table->not_redundant());
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the record is the metadata pseudo-record
 | 
						|
in the clustered index for instant ADD COLUMN (not other ALTER TABLE).
 | 
						|
@param[in]	rec	leaf page record
 | 
						|
@param[in]	comp	0 if ROW_FORMAT=REDUNDANT, else nonzero
 | 
						|
@return	whether the record is the metadata pseudo-record */
 | 
						|
inline bool rec_is_add_metadata(const rec_t* rec, ulint comp)
 | 
						|
{
 | 
						|
	bool is = rec_get_info_bits(rec, comp) == REC_INFO_MIN_REC_FLAG;
 | 
						|
	ut_ad(!is || !comp || rec_get_status(rec) == REC_STATUS_INSTANT);
 | 
						|
	return is;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the record is the metadata pseudo-record
 | 
						|
in the clustered index for instant ADD COLUMN (not other ALTER TABLE).
 | 
						|
@param[in]	rec	leaf page record
 | 
						|
@param[in]	index	index of the record
 | 
						|
@return	whether the record is the metadata pseudo-record */
 | 
						|
inline bool rec_is_add_metadata(const rec_t* rec, const dict_index_t& index)
 | 
						|
{
 | 
						|
	bool is = rec_is_add_metadata(rec, dict_table_is_comp(index.table));
 | 
						|
	ut_ad(!is || index.is_instant());
 | 
						|
	return is;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the record is the metadata pseudo-record
 | 
						|
in the clustered index for instant ALTER TABLE (not plain ADD COLUMN).
 | 
						|
@param[in]	rec	leaf page record
 | 
						|
@param[in]	comp	0 if ROW_FORMAT=REDUNDANT, else nonzero
 | 
						|
@return	whether the record is the ALTER TABLE metadata pseudo-record */
 | 
						|
inline bool rec_is_alter_metadata(const rec_t* rec, ulint comp)
 | 
						|
{
 | 
						|
	bool is = !(~rec_get_info_bits(rec, comp)
 | 
						|
		    & (REC_INFO_MIN_REC_FLAG | REC_INFO_DELETED_FLAG));
 | 
						|
	ut_ad(!is || rec_is_metadata(rec, comp));
 | 
						|
	return is;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if the record is the metadata pseudo-record
 | 
						|
in the clustered index for instant ALTER TABLE (not plain ADD COLUMN).
 | 
						|
@param[in]	rec	leaf page record
 | 
						|
@param[in]	index	index of the record
 | 
						|
@return	whether the record is the ALTER TABLE metadata pseudo-record */
 | 
						|
inline bool rec_is_alter_metadata(const rec_t* rec, const dict_index_t& index)
 | 
						|
{
 | 
						|
	bool is = rec_is_alter_metadata(rec, dict_table_is_comp(index.table));
 | 
						|
	ut_ad(!is || index.is_dummy || index.is_instant());
 | 
						|
	return is;
 | 
						|
}
 | 
						|
 | 
						|
/** Determine if a record is delete-marked (not a metadata pseudo-record).
 | 
						|
@param[in]	rec	record
 | 
						|
@param[in]	comp	nonzero if ROW_FORMAT!=REDUNDANT
 | 
						|
@return	whether the record is a delete-marked user record */
 | 
						|
inline bool rec_is_delete_marked(const rec_t* rec, ulint comp)
 | 
						|
{
 | 
						|
	return (rec_get_info_bits(rec, comp)
 | 
						|
		& (REC_INFO_MIN_REC_FLAG | REC_INFO_DELETED_FLAG))
 | 
						|
		== REC_INFO_DELETED_FLAG;
 | 
						|
}
 | 
						|
 | 
						|
/** Get the nth field from an index.
 | 
						|
@param[in]	rec	index record
 | 
						|
@param[in]	index	index
 | 
						|
@param[in]	offsets	rec_get_offsets(rec, index)
 | 
						|
@param[in]	n	field number
 | 
						|
@param[out]	len	length of the field in bytes, or UNIV_SQL_NULL
 | 
						|
@return a read-only copy of the index field */
 | 
						|
inline
 | 
						|
const byte*
 | 
						|
rec_get_nth_cfield(
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	const rec_offs*		offsets,
 | 
						|
	ulint			n,
 | 
						|
	ulint*			len)
 | 
						|
{
 | 
						|
	/* Because this function may be invoked by innobase_rec_to_mysql()
 | 
						|
	for reporting a duplicate key during ALTER TABLE or
 | 
						|
	CREATE UNIQUE INDEX, and in that case the rec omit the fixed-size
 | 
						|
	header of 5 or 6 bytes, the check
 | 
						|
	rec_offs_validate(rec, index, offsets) must be avoided here. */
 | 
						|
	if (!rec_offs_nth_default(offsets, n)) {
 | 
						|
		return rec_get_nth_field(rec, offsets, n, len);
 | 
						|
	}
 | 
						|
	return index->instant_field_value(n, len);
 | 
						|
}
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
Gets the physical size of a field.
 | 
						|
@return length of field */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_offs_nth_size(
 | 
						|
/*==============*/
 | 
						|
	const rec_offs*	offsets,/*!< in: array returned by rec_get_offsets() */
 | 
						|
	ulint		n)	/*!< in: nth field */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
 | 
						|
/******************************************************//**
 | 
						|
Returns the number of extern bits set in a record.
 | 
						|
@return number of externally stored fields */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_offs_n_extern(
 | 
						|
/*==============*/
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/**********************************************************//**
 | 
						|
The following function returns the data size of an old-style physical
 | 
						|
record, that is the sum of field lengths. SQL null fields
 | 
						|
are counted as length 0 fields. The value returned by the function
 | 
						|
is the distance from record origin to record end in bytes.
 | 
						|
@return size */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_data_size_old(
 | 
						|
/*==================*/
 | 
						|
	const rec_t*	rec)	/*!< in: physical record */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/**********************************************************//**
 | 
						|
The following function sets the number of allocated elements
 | 
						|
for an array of offsets. */
 | 
						|
UNIV_INLINE
 | 
						|
void
 | 
						|
rec_offs_set_n_alloc(
 | 
						|
/*=================*/
 | 
						|
	rec_offs*offsets,	/*!< out: array for rec_get_offsets(),
 | 
						|
				must be allocated */
 | 
						|
	ulint	n_alloc)	/*!< in: number of elements */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
#define rec_offs_init(offsets) \
 | 
						|
	rec_offs_set_n_alloc(offsets, (sizeof offsets) / sizeof *offsets)
 | 
						|
/**********************************************************//**
 | 
						|
The following function returns the data size of a physical
 | 
						|
record, that is the sum of field lengths. SQL null fields
 | 
						|
are counted as length 0 fields. The value returned by the function
 | 
						|
is the distance from record origin to record end in bytes.
 | 
						|
@return size */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_offs_data_size(
 | 
						|
/*===============*/
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/**********************************************************//**
 | 
						|
Returns the total size of record minus data size of record.
 | 
						|
The value returned by the function is the distance from record
 | 
						|
start to record origin in bytes.
 | 
						|
@return size */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_offs_extra_size(
 | 
						|
/*================*/
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/**********************************************************//**
 | 
						|
Returns the total size of a physical record.
 | 
						|
@return size */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_offs_size(
 | 
						|
/*==========*/
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
#ifdef UNIV_DEBUG
 | 
						|
/**********************************************************//**
 | 
						|
Returns a pointer to the start of the record.
 | 
						|
@return pointer to start */
 | 
						|
UNIV_INLINE
 | 
						|
byte*
 | 
						|
rec_get_start(
 | 
						|
/*==========*/
 | 
						|
	const rec_t*	rec,	/*!< in: pointer to record */
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/**********************************************************//**
 | 
						|
Returns a pointer to the end of the record.
 | 
						|
@return pointer to end */
 | 
						|
UNIV_INLINE
 | 
						|
byte*
 | 
						|
rec_get_end(
 | 
						|
/*========*/
 | 
						|
	const rec_t*	rec,	/*!< in: pointer to record */
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
#else /* UNIV_DEBUG */
 | 
						|
# define rec_get_start(rec, offsets) ((rec) - rec_offs_extra_size(offsets))
 | 
						|
# define rec_get_end(rec, offsets) ((rec) + rec_offs_data_size(offsets))
 | 
						|
#endif /* UNIV_DEBUG */
 | 
						|
 | 
						|
/** Copy a physical record to a buffer.
 | 
						|
@param[in]	buf	buffer
 | 
						|
@param[in]	rec	physical record
 | 
						|
@param[in]	offsets	array returned by rec_get_offsets()
 | 
						|
@return pointer to the origin of the copy */
 | 
						|
UNIV_INLINE
 | 
						|
rec_t*
 | 
						|
rec_copy(
 | 
						|
	void*		buf,
 | 
						|
	const rec_t*	rec,
 | 
						|
	const rec_offs*	offsets);
 | 
						|
 | 
						|
/** Determine the size of a data tuple prefix in a temporary file.
 | 
						|
@tparam redundant_temp whether to use the ROW_FORMAT=REDUNDANT format
 | 
						|
@param[in]	index		clustered or secondary index
 | 
						|
@param[in]	fields		data fields
 | 
						|
@param[in]	n_fields	number of data fields
 | 
						|
@param[out]	extra		record header size
 | 
						|
@param[in]	status		REC_STATUS_ORDINARY or REC_STATUS_INSTANT
 | 
						|
@return	total size, in bytes */
 | 
						|
template<bool redundant_temp>
 | 
						|
ulint
 | 
						|
rec_get_converted_size_temp(
 | 
						|
	const dict_index_t*	index,
 | 
						|
	const dfield_t*		fields,
 | 
						|
	ulint			n_fields,
 | 
						|
	ulint*			extra,
 | 
						|
	rec_comp_status_t	status = REC_STATUS_ORDINARY)
 | 
						|
	MY_ATTRIBUTE((warn_unused_result, nonnull));
 | 
						|
 | 
						|
/** Determine the offset to each field in temporary file.
 | 
						|
@param[in]	rec	temporary file record
 | 
						|
@param[in]	index	index of that the record belongs to
 | 
						|
@param[in,out]	offsets	offsets to the fields; in: rec_offs_n_fields(offsets)
 | 
						|
@param[in]	n_core	number of core fields (index->n_core_fields)
 | 
						|
@param[in]	def_val	default values for non-core fields
 | 
						|
@param[in]	status	REC_STATUS_ORDINARY or REC_STATUS_INSTANT */
 | 
						|
void
 | 
						|
rec_init_offsets_temp(
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	rec_offs*		offsets,
 | 
						|
	ulint			n_core,
 | 
						|
	const dict_col_t::def_t*def_val,
 | 
						|
	rec_comp_status_t	status = REC_STATUS_ORDINARY)
 | 
						|
	MY_ATTRIBUTE((nonnull(1,2,3)));
 | 
						|
/** Determine the offset to each field in temporary file.
 | 
						|
@param[in]	rec	temporary file record
 | 
						|
@param[in]	index	index of that the record belongs to
 | 
						|
@param[in,out]	offsets	offsets to the fields; in: rec_offs_n_fields(offsets)
 | 
						|
*/
 | 
						|
void
 | 
						|
rec_init_offsets_temp(
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	rec_offs*		offsets)
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
 | 
						|
/** Convert a data tuple prefix to the temporary file format.
 | 
						|
@tparam redundant_temp whether to use the ROW_FORMAT=REDUNDANT format
 | 
						|
@param[out]	rec		record in temporary file format
 | 
						|
@param[in]	index		clustered or secondary index
 | 
						|
@param[in]	fields		data fields
 | 
						|
@param[in]	n_fields	number of data fields
 | 
						|
@param[in]	status		REC_STATUS_ORDINARY or REC_STATUS_INSTANT */
 | 
						|
template<bool redundant_temp>
 | 
						|
void
 | 
						|
rec_convert_dtuple_to_temp(
 | 
						|
	rec_t*			rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	const dfield_t*		fields,
 | 
						|
	ulint			n_fields,
 | 
						|
	rec_comp_status_t	status = REC_STATUS_ORDINARY)
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
 | 
						|
/**************************************************************//**
 | 
						|
Copies the first n fields of a physical record to a new physical record in
 | 
						|
a buffer.
 | 
						|
@return own: copied record */
 | 
						|
rec_t*
 | 
						|
rec_copy_prefix_to_buf(
 | 
						|
/*===================*/
 | 
						|
	const rec_t*		rec,		/*!< in: physical record */
 | 
						|
	const dict_index_t*	index,		/*!< in: record descriptor */
 | 
						|
	ulint			n_fields,	/*!< in: number of fields
 | 
						|
						to copy */
 | 
						|
	byte**			buf,		/*!< in/out: memory buffer
 | 
						|
						for the copied prefix,
 | 
						|
						or NULL */
 | 
						|
	ulint*			buf_size)	/*!< in/out: buffer size */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/*********************************************************//**
 | 
						|
Builds a physical record out of a data tuple and
 | 
						|
stores it into the given buffer.
 | 
						|
@return pointer to the origin of physical record */
 | 
						|
rec_t*
 | 
						|
rec_convert_dtuple_to_rec(
 | 
						|
/*======================*/
 | 
						|
	byte*			buf,	/*!< in: start address of the
 | 
						|
					physical record */
 | 
						|
	const dict_index_t*	index,	/*!< in: record descriptor */
 | 
						|
	const dtuple_t*		dtuple,	/*!< in: data tuple */
 | 
						|
	ulint			n_ext)	/*!< in: number of
 | 
						|
					externally stored columns */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result));
 | 
						|
/**********************************************************//**
 | 
						|
Returns the extra size of an old-style physical record if we know its
 | 
						|
data size and number of fields.
 | 
						|
@return extra size */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_converted_extra_size(
 | 
						|
/*=========================*/
 | 
						|
	ulint	data_size,	/*!< in: data size */
 | 
						|
	ulint	n_fields,	/*!< in: number of fields */
 | 
						|
	ulint	n_ext)		/*!< in: number of externally stored columns */
 | 
						|
	MY_ATTRIBUTE((const));
 | 
						|
/**********************************************************//**
 | 
						|
Determines the size of a data tuple prefix in ROW_FORMAT=COMPACT.
 | 
						|
@return total size */
 | 
						|
ulint
 | 
						|
rec_get_converted_size_comp_prefix(
 | 
						|
/*===============================*/
 | 
						|
	const dict_index_t*	index,	/*!< in: record descriptor */
 | 
						|
	const dfield_t*		fields,	/*!< in: array of data fields */
 | 
						|
	ulint			n_fields,/*!< in: number of data fields */
 | 
						|
	ulint*			extra)	/*!< out: extra size */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result, nonnull(1,2)));
 | 
						|
 | 
						|
/** Determine the size of a record in ROW_FORMAT=COMPACT.
 | 
						|
@param[in]	index		record descriptor. dict_table_is_comp()
 | 
						|
				is assumed to hold, even if it doesn't
 | 
						|
@param[in]	tuple		logical record
 | 
						|
@param[out]	extra		extra size
 | 
						|
@return total size */
 | 
						|
ulint
 | 
						|
rec_get_converted_size_comp(
 | 
						|
	const dict_index_t*	index,
 | 
						|
	const dtuple_t*		tuple,
 | 
						|
	ulint*			extra)
 | 
						|
	MY_ATTRIBUTE((nonnull(1,2)));
 | 
						|
 | 
						|
/**********************************************************//**
 | 
						|
The following function returns the size of a data tuple when converted to
 | 
						|
a physical record.
 | 
						|
@return size */
 | 
						|
UNIV_INLINE
 | 
						|
ulint
 | 
						|
rec_get_converted_size(
 | 
						|
/*===================*/
 | 
						|
	dict_index_t*	index,	/*!< in: record descriptor */
 | 
						|
	const dtuple_t*	dtuple,	/*!< in: data tuple */
 | 
						|
	ulint		n_ext)	/*!< in: number of externally stored columns */
 | 
						|
	MY_ATTRIBUTE((warn_unused_result, nonnull));
 | 
						|
/** Copy the first n fields of a (copy of a) physical record to a data tuple.
 | 
						|
The fields are copied into the memory heap.
 | 
						|
@param[out]	tuple		data tuple
 | 
						|
@param[in]	rec		index record, or a copy thereof
 | 
						|
@param[in]	index		index of rec
 | 
						|
@param[in]	n_core		index->n_core_fields at the time rec was
 | 
						|
				copied, or 0 if non-leaf page record
 | 
						|
@param[in]	n_fields	number of fields to copy
 | 
						|
@param[in,out]	heap		memory heap */
 | 
						|
void
 | 
						|
rec_copy_prefix_to_dtuple(
 | 
						|
	dtuple_t*		tuple,
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index,
 | 
						|
	ulint			n_core,
 | 
						|
	ulint			n_fields,
 | 
						|
	mem_heap_t*		heap)
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/***************************************************************//**
 | 
						|
Validates the consistency of a physical record.
 | 
						|
@return TRUE if ok */
 | 
						|
ibool
 | 
						|
rec_validate(
 | 
						|
/*=========*/
 | 
						|
	const rec_t*	rec,	/*!< in: physical record */
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/***************************************************************//**
 | 
						|
Prints an old-style physical record. */
 | 
						|
void
 | 
						|
rec_print_old(
 | 
						|
/*==========*/
 | 
						|
	FILE*		file,	/*!< in: file where to print */
 | 
						|
	const rec_t*	rec)	/*!< in: physical record */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/***************************************************************//**
 | 
						|
Prints a spatial index record. */
 | 
						|
void
 | 
						|
rec_print_mbr_rec(
 | 
						|
/*==========*/
 | 
						|
	FILE*		file,	/*!< in: file where to print */
 | 
						|
	const rec_t*	rec,	/*!< in: physical record */
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/***************************************************************//**
 | 
						|
Prints a physical record. */
 | 
						|
void
 | 
						|
rec_print_new(
 | 
						|
/*==========*/
 | 
						|
	FILE*		file,	/*!< in: file where to print */
 | 
						|
	const rec_t*	rec,	/*!< in: physical record */
 | 
						|
	const rec_offs*	offsets)/*!< in: array returned by rec_get_offsets() */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
/***************************************************************//**
 | 
						|
Prints a physical record. */
 | 
						|
void
 | 
						|
rec_print(
 | 
						|
/*======*/
 | 
						|
	FILE*			file,	/*!< in: file where to print */
 | 
						|
	const rec_t*		rec,	/*!< in: physical record */
 | 
						|
	const dict_index_t*	index)	/*!< in: record descriptor */
 | 
						|
	MY_ATTRIBUTE((nonnull));
 | 
						|
 | 
						|
/** Pretty-print a record.
 | 
						|
@param[in,out]	o	output stream
 | 
						|
@param[in]	rec	physical record
 | 
						|
@param[in]	info	rec_get_info_bits(rec)
 | 
						|
@param[in]	offsets	rec_get_offsets(rec) */
 | 
						|
void
 | 
						|
rec_print(
 | 
						|
	std::ostream&	o,
 | 
						|
	const rec_t*	rec,
 | 
						|
	ulint		info,
 | 
						|
	const rec_offs*	offsets);
 | 
						|
 | 
						|
/** Wrapper for pretty-printing a record */
 | 
						|
struct rec_index_print
 | 
						|
{
 | 
						|
	/** Constructor */
 | 
						|
	rec_index_print(const rec_t* rec, const dict_index_t* index) :
 | 
						|
		m_rec(rec), m_index(index)
 | 
						|
	{}
 | 
						|
 | 
						|
	/** Record */
 | 
						|
	const rec_t*		m_rec;
 | 
						|
	/** Index */
 | 
						|
	const dict_index_t*	m_index;
 | 
						|
};
 | 
						|
 | 
						|
/** Display a record.
 | 
						|
@param[in,out]	o	output stream
 | 
						|
@param[in]	r	record to display
 | 
						|
@return	the output stream */
 | 
						|
std::ostream&
 | 
						|
operator<<(std::ostream& o, const rec_index_print& r);
 | 
						|
 | 
						|
/** Wrapper for pretty-printing a record */
 | 
						|
struct rec_offsets_print
 | 
						|
{
 | 
						|
	/** Constructor */
 | 
						|
	rec_offsets_print(const rec_t* rec, const rec_offs* offsets) :
 | 
						|
		m_rec(rec), m_offsets(offsets)
 | 
						|
	{}
 | 
						|
 | 
						|
	/** Record */
 | 
						|
	const rec_t*		m_rec;
 | 
						|
	/** Offsets to each field */
 | 
						|
	const rec_offs*		m_offsets;
 | 
						|
};
 | 
						|
 | 
						|
/** Display a record.
 | 
						|
@param[in,out]	o	output stream
 | 
						|
@param[in]	r	record to display
 | 
						|
@return	the output stream */
 | 
						|
ATTRIBUTE_COLD
 | 
						|
std::ostream&
 | 
						|
operator<<(std::ostream& o, const rec_offsets_print& r);
 | 
						|
 | 
						|
/** Pretty-printer of records and tuples */
 | 
						|
class rec_printer : public std::ostringstream {
 | 
						|
public:
 | 
						|
	/** Construct a pretty-printed record.
 | 
						|
	@param rec	record with header
 | 
						|
	@param offsets	rec_get_offsets(rec, ...) */
 | 
						|
	ATTRIBUTE_COLD
 | 
						|
	rec_printer(const rec_t* rec, const rec_offs* offsets)
 | 
						|
		:
 | 
						|
		std::ostringstream ()
 | 
						|
	{
 | 
						|
		rec_print(*this, rec,
 | 
						|
			  rec_get_info_bits(rec, rec_offs_comp(offsets)),
 | 
						|
			  offsets);
 | 
						|
	}
 | 
						|
 | 
						|
	/** Construct a pretty-printed record.
 | 
						|
	@param rec record, possibly lacking header
 | 
						|
	@param info rec_get_info_bits(rec)
 | 
						|
	@param offsets rec_get_offsets(rec, ...) */
 | 
						|
	ATTRIBUTE_COLD
 | 
						|
	rec_printer(const rec_t* rec, ulint info, const rec_offs* offsets)
 | 
						|
		:
 | 
						|
		std::ostringstream ()
 | 
						|
	{
 | 
						|
		rec_print(*this, rec, info, offsets);
 | 
						|
	}
 | 
						|
 | 
						|
	/** Construct a pretty-printed tuple.
 | 
						|
	@param tuple	data tuple */
 | 
						|
	ATTRIBUTE_COLD
 | 
						|
	rec_printer(const dtuple_t* tuple)
 | 
						|
		:
 | 
						|
		std::ostringstream ()
 | 
						|
	{
 | 
						|
		dtuple_print(*this, tuple);
 | 
						|
	}
 | 
						|
 | 
						|
	/** Construct a pretty-printed tuple.
 | 
						|
	@param field	array of data tuple fields
 | 
						|
	@param n	number of fields */
 | 
						|
	ATTRIBUTE_COLD
 | 
						|
	rec_printer(const dfield_t* field, ulint n)
 | 
						|
		:
 | 
						|
		std::ostringstream ()
 | 
						|
	{
 | 
						|
		dfield_print(*this, field, n);
 | 
						|
	}
 | 
						|
 | 
						|
	/** Destructor */
 | 
						|
	~rec_printer() override = default;
 | 
						|
 | 
						|
private:
 | 
						|
	/** Copy constructor */
 | 
						|
	rec_printer(const rec_printer& other);
 | 
						|
	/** Assignment operator */
 | 
						|
	rec_printer& operator=(const rec_printer& other);
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
# ifdef UNIV_DEBUG
 | 
						|
/** Read the DB_TRX_ID of a clustered index record.
 | 
						|
@param[in]	rec	clustered index record
 | 
						|
@param[in]	index	clustered index
 | 
						|
@return the value of DB_TRX_ID */
 | 
						|
trx_id_t
 | 
						|
rec_get_trx_id(
 | 
						|
	const rec_t*		rec,
 | 
						|
	const dict_index_t*	index)
 | 
						|
	MY_ATTRIBUTE((nonnull, warn_unused_result));
 | 
						|
# endif /* UNIV_DEBUG */
 | 
						|
 | 
						|
/* Maximum lengths for the data in a physical record if the offsets
 | 
						|
are given in one byte (resp. two byte) format. */
 | 
						|
#define REC_1BYTE_OFFS_LIMIT	0x7FUL
 | 
						|
#define REC_2BYTE_OFFS_LIMIT	0x7FFFUL
 | 
						|
 | 
						|
/* The data size of record must not be larger than this on
 | 
						|
REDUNDANT row format because we reserve two upmost bits in a
 | 
						|
two byte offset for special purposes */
 | 
						|
#define REDUNDANT_REC_MAX_DATA_SIZE    (16383)
 | 
						|
 | 
						|
/* The data size of record must be smaller than this on
 | 
						|
COMPRESSED row format because we reserve two upmost bits in a
 | 
						|
two byte offset for special purposes */
 | 
						|
#define COMPRESSED_REC_MAX_DATA_SIZE   (16384)
 | 
						|
 | 
						|
#ifdef WITH_WSREP
 | 
						|
int wsrep_rec_get_foreign_key(
 | 
						|
	byte 		*buf,     /* out: extracted key */
 | 
						|
	ulint 		*buf_len, /* in/out: length of buf */
 | 
						|
	const rec_t*	rec,	  /* in: physical record */
 | 
						|
	dict_index_t*	index_for,  /* in: index for foreign table */
 | 
						|
	dict_index_t*	index_ref,  /* in: index for referenced table */
 | 
						|
	ibool		new_protocol); /* in: protocol > 1 */
 | 
						|
#endif /* WITH_WSREP */
 | 
						|
 | 
						|
#include "rem0rec.inl"
 | 
						|
 | 
						|
#endif /* !UNIV_INNOCHECKSUM */
 | 
						|
#endif /* rem0rec_h */
 |