mirror of
https://github.com/MariaDB/server.git
synced 2025-01-19 13:32:33 +01:00
Merge mysql.com:/home/pz/mysql/mysql-4.0-root
into mysql.com:/home/pz/mysql/mysql-4.0
This commit is contained in:
commit
2440182b1e
79 changed files with 1631 additions and 542 deletions
|
@ -3554,10 +3554,13 @@ string) in it instead. (This behaviour can, however, be changed with the
|
||||||
-DDONT_USE_DEFAULT_FIELDS compile option.)
|
-DDONT_USE_DEFAULT_FIELDS compile option.)
|
||||||
|
|
||||||
@item
|
@item
|
||||||
MySQL allows you to store some wrong date values into
|
MySQL allows you to store some wrong date values into @code{DATE} and
|
||||||
@code{DATE} and @code{DATETIME} columns (like 2000-02-31 or 2000-02-00).
|
@code{DATETIME} columns (like 2000-02-31 or 2000-02-00). The idea is
|
||||||
If the date is totally wrong, MySQL Server will store the special
|
that it's not the SQL server job to vaildate date. If MySQL can store a
|
||||||
0000-00-00 date value in the column.
|
date and retrieve exactly the same date, then MySQL will store the
|
||||||
|
date. If the date is totally wrong (outside of MySQL abilty to store
|
||||||
|
it), MySQL Server will store the special 0000-00-00 date value in the
|
||||||
|
column.
|
||||||
|
|
||||||
@item
|
@item
|
||||||
If you set an @code{ENUM} column to an unsupported value, it will be set to
|
If you set an @code{ENUM} column to an unsupported value, it will be set to
|
||||||
|
@ -8170,6 +8173,9 @@ Renamed mysqld startup options @code{--skip-locking} to
|
||||||
@code{--skip-external-locking} and @code{--enable-locking} to
|
@code{--skip-external-locking} and @code{--enable-locking} to
|
||||||
@code{--external-locking}.
|
@code{--external-locking}.
|
||||||
@item
|
@item
|
||||||
|
mysqld now has the option @code{--temp-pool} enabled by default as this
|
||||||
|
gives better performance with some OS.
|
||||||
|
@item
|
||||||
@code{DOUBLE} and @code{FLOAT} columns now honour the
|
@code{DOUBLE} and @code{FLOAT} columns now honour the
|
||||||
@code{UNSIGNED} flag on storage (before, @code{UNSIGNED} was ignored for
|
@code{UNSIGNED} flag on storage (before, @code{UNSIGNED} was ignored for
|
||||||
these columns).
|
these columns).
|
||||||
|
@ -32555,6 +32561,31 @@ mysql> SELECT WEEK('1998-12-31',1);
|
||||||
Note: in Version 4.0, @code{WEEK(#,0)} was changed to match the
|
Note: in Version 4.0, @code{WEEK(#,0)} was changed to match the
|
||||||
calendar in the USA.
|
calendar in the USA.
|
||||||
|
|
||||||
|
Note that if a week is the last week of the previous year, MySQL will
|
||||||
|
return 0:
|
||||||
|
|
||||||
|
@example
|
||||||
|
mysql> SELECT YEAR('2000-01-01'), WEEK('2000-01-01',0);
|
||||||
|
-> 2000, 0
|
||||||
|
@end example
|
||||||
|
|
||||||
|
One could argue that MySQL should return @code{52} for the @code{WEEK()}
|
||||||
|
function as the given date is actually the 52 second week of 1999. We
|
||||||
|
decided to return 0 instead as we want the function to return 'the week
|
||||||
|
number in the given year'. This makes the usage of the @code{WEEK()}
|
||||||
|
function reliable when combined with other functions that extracts a
|
||||||
|
date part from a date.
|
||||||
|
|
||||||
|
If you would prefer to know the correct year-week, then you should use
|
||||||
|
the @code{YEARWEEK()} function instead:
|
||||||
|
|
||||||
|
@example
|
||||||
|
mysql> SELECT YEARWEEK('2000-01-01');
|
||||||
|
-> 199952
|
||||||
|
mysql> SELECT MID(YEARWEEK('2000-01-01'),5,2);
|
||||||
|
-> 52
|
||||||
|
@end example
|
||||||
|
|
||||||
@findex YEAR()
|
@findex YEAR()
|
||||||
@item YEAR(date)
|
@item YEAR(date)
|
||||||
Returns the year for @code{date}, in the range @code{1000} to @code{9999}:
|
Returns the year for @code{date}, in the range @code{1000} to @code{9999}:
|
||||||
|
@ -32576,6 +32607,10 @@ mysql> SELECT YEARWEEK('1987-01-01');
|
||||||
-> 198653
|
-> 198653
|
||||||
@end example
|
@end example
|
||||||
|
|
||||||
|
Note that the week number is different from what the @code{WEEK()} function
|
||||||
|
would return (@code{0}) as the week function returns the week in the
|
||||||
|
context of the given year.
|
||||||
|
|
||||||
@findex HOUR()
|
@findex HOUR()
|
||||||
@item HOUR(time)
|
@item HOUR(time)
|
||||||
Returns the hour for @code{time}, in the range @code{0} to @code{23}:
|
Returns the hour for @code{time}, in the range @code{0} to @code{23}:
|
||||||
|
@ -48180,13 +48215,33 @@ mysql> SELECT idate FROM tbl_name WHERE STRCMP(idate,'19970505')=0;
|
||||||
a string and performs a string comparison. It does not convert
|
a string and performs a string comparison. It does not convert
|
||||||
@code{'19970505'} to a date and perform a date comparison.
|
@code{'19970505'} to a date and perform a date comparison.
|
||||||
|
|
||||||
Note that MySQL does no checking whether the date is
|
Note that MySQL does very limited checking whether the date is
|
||||||
correct. If you store an incorrect date, such as @code{'1998-2-31'}, the
|
correct. If you store an incorrect date, such as @code{'1998-2-31'}, the
|
||||||
wrong date will be stored. If the date cannot be converted to any reasonable
|
wrong date will be stored.
|
||||||
value, a @code{0} is stored in the @code{DATE} field. This is mainly a speed
|
|
||||||
issue and we think it is up to the application to check the dates, and not
|
|
||||||
the server.
|
|
||||||
|
|
||||||
|
Because MySQL packs dates for storage, it can't store any given date as
|
||||||
|
it would not fit onto the result buffer. The rules for accepting a date
|
||||||
|
are:
|
||||||
|
@itemize @bullet
|
||||||
|
@item
|
||||||
|
If MySQL can store it and retrieve a date, the wrong date is accepted
|
||||||
|
for @code{DATE} and @code{DATETIME} columns.
|
||||||
|
@item
|
||||||
|
All days values between 0-31 are accepted for any date. This makes it
|
||||||
|
very convenient for web applications where you ask year, month and day
|
||||||
|
in 3 different fields.
|
||||||
|
@item
|
||||||
|
The day or month field may be zero. This is convenient if you want
|
||||||
|
to store a birthdate in a @code{DATE} column and you only know part
|
||||||
|
of the date.
|
||||||
|
@end itemize
|
||||||
|
|
||||||
|
If the date cannot be converted to any reasonable value, a @code{0} is
|
||||||
|
stored in the @code{DATE} field, which will be retrieved as
|
||||||
|
@code{0000-00-00}. This is both a speed and convinient issue as we
|
||||||
|
belive that the databases responsiblity etrive the same date you stored
|
||||||
|
(even if the data was not logicall correct in all cases). We think it is
|
||||||
|
up to the application to check the dates, and not the server.
|
||||||
|
|
||||||
@node Problems with NULL, Problems with alias, Using DATE, Query Issues
|
@node Problems with NULL, Problems with alias, Using DATE, Query Issues
|
||||||
@appendixsubsec Problems with @code{NULL} Values
|
@appendixsubsec Problems with @code{NULL} Values
|
||||||
|
@ -50167,6 +50222,9 @@ each individual 4.0.x release.
|
||||||
|
|
||||||
@itemize @bullet
|
@itemize @bullet
|
||||||
@item
|
@item
|
||||||
|
mysqld now has the option @code{--temp-pool} enabled by default as this
|
||||||
|
gives better performance with some OS.
|
||||||
|
@item
|
||||||
Big code cleanup in replication code.
|
Big code cleanup in replication code.
|
||||||
@item
|
@item
|
||||||
If one specifies @code{--code-file}, call @code{setrlmit()} to change allowed
|
If one specifies @code{--code-file}, call @code{setrlmit()} to change allowed
|
||||||
|
@ -50804,6 +50862,23 @@ not yet 100% confident in this code.
|
||||||
@appendixsubsec Changes in release 3.23.52
|
@appendixsubsec Changes in release 3.23.52
|
||||||
@itemize @bullet
|
@itemize @bullet
|
||||||
@item
|
@item
|
||||||
|
Fixed a security bug with empty db column in db table
|
||||||
|
@item
|
||||||
|
Changed initialisation of @code{RND()} to make it less predicatable.
|
||||||
|
@item
|
||||||
|
Fixed problem with @code{GROUP BY} on result with expression that created a
|
||||||
|
@code{BLOB} field.
|
||||||
|
@item
|
||||||
|
Fixed problem with privilege tables when downgrading from 4.0.2 to 3.23.
|
||||||
|
@item
|
||||||
|
Fixed thread bug in @code{SLAVE START}, @code{SLAVE STOP} and automatic repair
|
||||||
|
of MyISAM tables that could cause table cache to be corrupted.
|
||||||
|
@item
|
||||||
|
Fixed possible thread related key-cache-corruption problem with
|
||||||
|
@code{OPTIMIZE TABLE} and @code{REPAIR TABLE}.
|
||||||
|
@item
|
||||||
|
Added name of 'administrator command' logs.
|
||||||
|
@item
|
||||||
Fixed bug with creating an auto-increment value on second part of a
|
Fixed bug with creating an auto-increment value on second part of a
|
||||||
@code{UNIQUE()} key where first part could contain NULL values.
|
@code{UNIQUE()} key where first part could contain NULL values.
|
||||||
@item
|
@item
|
||||||
|
@ -50811,7 +50886,9 @@ Don't write slave-timeout reconnects to the error log.
|
||||||
@item
|
@item
|
||||||
Fixed bug with slave net read timeouting
|
Fixed bug with slave net read timeouting
|
||||||
@item
|
@item
|
||||||
Fixed bug in ALTERing TABLE of BDB type.
|
Fixed a core-dump bug with @code{MERGE} tables and @code{MAX()} function.
|
||||||
|
@item
|
||||||
|
Fixed bug in @code{ALTER TABLE} with BDB tables.
|
||||||
@item
|
@item
|
||||||
Fixed bug when logging @code{LOAD DATA INFILE} to binary log with no
|
Fixed bug when logging @code{LOAD DATA INFILE} to binary log with no
|
||||||
active database.
|
active database.
|
||||||
|
|
13
configure.in
13
configure.in
|
@ -1554,6 +1554,17 @@ AC_SUBST(TERMCAP_LIB)
|
||||||
#########################################################################
|
#########################################################################
|
||||||
|
|
||||||
dnl Checks for library functions.
|
dnl Checks for library functions.
|
||||||
|
|
||||||
|
#
|
||||||
|
# The following code disables intrinsic function support while we test for
|
||||||
|
# library functions. This is to avoid configure problems with Intel ecc
|
||||||
|
# compiler
|
||||||
|
|
||||||
|
ORG_CFLAGS="$CFLAGS"
|
||||||
|
if test "$GCC" != "yes"; then
|
||||||
|
AC_SYS_COMPILER_FLAG(-nolib_inline,nolib_inline,CFLAGS,[],[])
|
||||||
|
fi
|
||||||
|
|
||||||
AC_FUNC_MMAP
|
AC_FUNC_MMAP
|
||||||
AC_TYPE_SIGNAL
|
AC_TYPE_SIGNAL
|
||||||
MYSQL_TYPE_QSORT
|
MYSQL_TYPE_QSORT
|
||||||
|
@ -1576,6 +1587,8 @@ AC_CHECK_FUNCS(alarm bmove \
|
||||||
pthread_condattr_create rwlock_init pthread_rwlock_rdlock \
|
pthread_condattr_create rwlock_init pthread_rwlock_rdlock \
|
||||||
fchmod getpass getpassphrase initgroups mlockall)
|
fchmod getpass getpassphrase initgroups mlockall)
|
||||||
|
|
||||||
|
CFLAGS="$ORG_CFLAGS"
|
||||||
|
|
||||||
# Sanity check: We chould not have any fseeko symbol unless
|
# Sanity check: We chould not have any fseeko symbol unless
|
||||||
# large_file_support=yes
|
# large_file_support=yes
|
||||||
AC_CHECK_FUNCS(fseeko,
|
AC_CHECK_FUNCS(fseeko,
|
||||||
|
|
|
@ -121,14 +121,6 @@
|
||||||
/* #define _AIX32_CURSES */ /* XXX: this breaks AIX 4.3.3 (others?). */
|
/* #define _AIX32_CURSES */ /* XXX: this breaks AIX 4.3.3 (others?). */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef __QNXNTO__
|
|
||||||
#define HAVE_ERRNO_AS_DEFINE
|
|
||||||
#define HAVE_FCNTL_LOCK
|
|
||||||
#undef HAVE_SYS_UN_H
|
|
||||||
#undef HAVE_FINITE
|
|
||||||
#undef HAVE_RINT
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef HAVE_BROKEN_SNPRINTF /* HPUX 10.20 don't have this defined */
|
#ifdef HAVE_BROKEN_SNPRINTF /* HPUX 10.20 don't have this defined */
|
||||||
#undef HAVE_SNPRINTF
|
#undef HAVE_SNPRINTF
|
||||||
#endif
|
#endif
|
||||||
|
@ -253,6 +245,17 @@
|
||||||
#define setrlimit cma_setrlimit64
|
#define setrlimit cma_setrlimit64
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef __QNXNTO__
|
||||||
|
/* This has to be after include limits.h */
|
||||||
|
#define HAVE_ERRNO_AS_DEFINE
|
||||||
|
#define HAVE_FCNTL_LOCK
|
||||||
|
#undef HAVE_SYS_UN_H
|
||||||
|
#undef HAVE_FINITE
|
||||||
|
#undef HAVE_RINT
|
||||||
|
#undef LONGLONG_MIN /* These get wrongly defined in QNX 6.2 */
|
||||||
|
#undef LONGLONG_MAX /* standard system library 'limits.h' */
|
||||||
|
#endif
|
||||||
|
|
||||||
/* We can not live without the following defines */
|
/* We can not live without the following defines */
|
||||||
|
|
||||||
#define USE_MYFUNC 1 /* Must use syscall indirection */
|
#define USE_MYFUNC 1 /* Must use syscall indirection */
|
||||||
|
@ -548,11 +551,6 @@ extern double my_atof(const char*);
|
||||||
#define HAVE_LONG_LONG 1
|
#define HAVE_LONG_LONG 1
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef __QNXNTO__
|
|
||||||
#undef LONGLONG_MIN /* These get wrongly defined in QNX 6.2 */
|
|
||||||
#undef LONGLONG_MAX /* standard system library 'limits.h' */
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(HAVE_LONG_LONG) && !defined(LONGLONG_MIN)
|
#if defined(HAVE_LONG_LONG) && !defined(LONGLONG_MIN)
|
||||||
#define LONGLONG_MIN ((long long) 0x8000000000000000LL)
|
#define LONGLONG_MIN ((long long) 0x8000000000000000LL)
|
||||||
#define LONGLONG_MAX ((long long) 0x7FFFFFFFFFFFFFFFLL)
|
#define LONGLONG_MAX ((long long) 0x7FFFFFFFFFFFFFFFLL)
|
||||||
|
|
|
@ -430,8 +430,9 @@ struct tm *localtime_r(const time_t *clock, struct tm *res);
|
||||||
|
|
||||||
#if defined(HPUX) && !defined(DONT_REMAP_PTHREAD_FUNCTIONS)
|
#if defined(HPUX) && !defined(DONT_REMAP_PTHREAD_FUNCTIONS)
|
||||||
#undef pthread_cond_timedwait
|
#undef pthread_cond_timedwait
|
||||||
|
#undef pthread_mutex_trylock
|
||||||
#define pthread_cond_timedwait(a,b,c) my_pthread_cond_timedwait((a),(b),(c))
|
#define pthread_cond_timedwait(a,b,c) my_pthread_cond_timedwait((a),(b),(c))
|
||||||
#define pthread_mutex_trylock(a) my_pthread_mutex_trylock((a))
|
#define pthread_mutex_trylock(a) my_pthread_mutex_trylock((a))
|
||||||
int my_pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex,
|
int my_pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex,
|
||||||
struct timespec *abstime);
|
struct timespec *abstime);
|
||||||
int my_pthread_mutex_trylock(pthread_mutex_t *mutex);
|
int my_pthread_mutex_trylock(pthread_mutex_t *mutex);
|
||||||
|
|
|
@ -35,8 +35,6 @@ C_MODE_START
|
||||||
#ifdef HAVE_SEMAPHORE_H
|
#ifdef HAVE_SEMAPHORE_H
|
||||||
#include <semaphore.h>
|
#include <semaphore.h>
|
||||||
#elif defined(__bsdi__)
|
#elif defined(__bsdi__)
|
||||||
#include <sys/errno.h>
|
|
||||||
#else
|
|
||||||
#ifdef __WIN__
|
#ifdef __WIN__
|
||||||
typedef HANDLE sem_t;
|
typedef HANDLE sem_t;
|
||||||
#else
|
#else
|
||||||
|
@ -45,7 +43,7 @@ typedef struct {
|
||||||
pthread_cond_t cond;
|
pthread_cond_t cond;
|
||||||
uint count;
|
uint count;
|
||||||
} sem_t;
|
} sem_t;
|
||||||
#endif
|
#endif /* __WIN__ */
|
||||||
|
|
||||||
int sem_init(sem_t * sem, int pshared, unsigned int value);
|
int sem_init(sem_t * sem, int pshared, unsigned int value);
|
||||||
int sem_destroy(sem_t * sem);
|
int sem_destroy(sem_t * sem);
|
||||||
|
@ -55,7 +53,7 @@ int sem_post(sem_t * sem);
|
||||||
int sem_post_multiple(sem_t * sem, unsigned int count);
|
int sem_post_multiple(sem_t * sem, unsigned int count);
|
||||||
int sem_getvalue(sem_t * sem, unsigned int * sval);
|
int sem_getvalue(sem_t * sem, unsigned int * sval);
|
||||||
|
|
||||||
#endif
|
#endif /* !__bsdi__ */
|
||||||
|
|
||||||
C_MODE_END
|
C_MODE_END
|
||||||
#endif /* !_my_semaphore_h_ */
|
#endif /* !_my_semaphore_h_ */
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
#define MYSQL_VERSION_ID @MYSQL_VERSION_ID@
|
#define MYSQL_VERSION_ID @MYSQL_VERSION_ID@
|
||||||
#define MYSQL_PORT @MYSQL_TCP_PORT@
|
#define MYSQL_PORT @MYSQL_TCP_PORT@
|
||||||
#define MYSQL_UNIX_ADDR "@MYSQL_UNIX_ADDR@"
|
#define MYSQL_UNIX_ADDR "@MYSQL_UNIX_ADDR@"
|
||||||
|
#define MYSQL_CONFIG_NAME "my"
|
||||||
|
|
||||||
/* mysqld compile time options */
|
/* mysqld compile time options */
|
||||||
#ifndef MYSQL_CHARSET
|
#ifndef MYSQL_CHARSET
|
||||||
|
|
|
@ -808,7 +808,7 @@ btr_cur_optimistic_insert(
|
||||||
|
|
||||||
if (!dtuple_check_typed_no_assert(entry)) {
|
if (!dtuple_check_typed_no_assert(entry)) {
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
"InnoDB: Error in a tuple to insert into table %lu index %lu\n",
|
"InnoDB: Error in a tuple to insert into table %lu index %s\n",
|
||||||
index->table_name, index->name);
|
index->table_name, index->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1213,6 +1213,8 @@ btr_cur_parse_update_in_place(
|
||||||
rec_offset = mach_read_from_2(ptr);
|
rec_offset = mach_read_from_2(ptr);
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
|
|
||||||
|
ut_a(rec_offset <= UNIV_PAGE_SIZE);
|
||||||
|
|
||||||
heap = mem_heap_create(256);
|
heap = mem_heap_create(256);
|
||||||
|
|
||||||
ptr = row_upd_index_parse(ptr, end_ptr, heap, &update);
|
ptr = row_upd_index_parse(ptr, end_ptr, heap, &update);
|
||||||
|
@ -1977,6 +1979,8 @@ btr_cur_parse_del_mark_set_clust_rec(
|
||||||
offset = mach_read_from_2(ptr);
|
offset = mach_read_from_2(ptr);
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
|
|
||||||
|
ut_a(offset <= UNIV_PAGE_SIZE);
|
||||||
|
|
||||||
if (page) {
|
if (page) {
|
||||||
rec = page + offset;
|
rec = page + offset;
|
||||||
|
|
||||||
|
@ -2127,6 +2131,8 @@ btr_cur_parse_del_mark_set_sec_rec(
|
||||||
offset = mach_read_from_2(ptr);
|
offset = mach_read_from_2(ptr);
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
|
|
||||||
|
ut_a(offset <= UNIV_PAGE_SIZE);
|
||||||
|
|
||||||
if (page) {
|
if (page) {
|
||||||
rec = page + offset;
|
rec = page + offset;
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,7 @@ Created 2/17/1996 Heikki Tuuri
|
||||||
#include "btr0cur.h"
|
#include "btr0cur.h"
|
||||||
#include "btr0pcur.h"
|
#include "btr0pcur.h"
|
||||||
#include "btr0btr.h"
|
#include "btr0btr.h"
|
||||||
|
#include "ha0ha.h"
|
||||||
|
|
||||||
ulint btr_search_n_succ = 0;
|
ulint btr_search_n_succ = 0;
|
||||||
ulint btr_search_n_hash_fail = 0;
|
ulint btr_search_n_hash_fail = 0;
|
||||||
|
|
|
@ -286,7 +286,7 @@ buf_page_print(
|
||||||
ut_print_timestamp(stderr);
|
ut_print_timestamp(stderr);
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
" InnoDB: Page dump in ascii and hex (%u bytes):\n%s",
|
" InnoDB: Page dump in ascii and hex (%u bytes):\n%s",
|
||||||
UNIV_PAGE_SIZE, buf);
|
(ulint)UNIV_PAGE_SIZE, buf);
|
||||||
fprintf(stderr, "InnoDB: End of page dump\n");
|
fprintf(stderr, "InnoDB: End of page dump\n");
|
||||||
|
|
||||||
mem_free(buf);
|
mem_free(buf);
|
||||||
|
@ -1707,10 +1707,11 @@ buf_print(void)
|
||||||
|
|
||||||
mutex_enter(&(buf_pool->mutex));
|
mutex_enter(&(buf_pool->mutex));
|
||||||
|
|
||||||
printf("LRU len %lu \n", UT_LIST_GET_LEN(buf_pool->LRU));
|
|
||||||
printf("free len %lu \n", UT_LIST_GET_LEN(buf_pool->free));
|
|
||||||
printf("flush len %lu \n", UT_LIST_GET_LEN(buf_pool->flush_list));
|
|
||||||
printf("buf_pool size %lu \n", size);
|
printf("buf_pool size %lu \n", size);
|
||||||
|
printf("database pages %lu \n", UT_LIST_GET_LEN(buf_pool->LRU));
|
||||||
|
printf("free pages %lu \n", UT_LIST_GET_LEN(buf_pool->free));
|
||||||
|
printf("modified database pages %lu \n",
|
||||||
|
UT_LIST_GET_LEN(buf_pool->flush_list));
|
||||||
|
|
||||||
printf("n pending reads %lu \n", buf_pool->n_pend_reads);
|
printf("n pending reads %lu \n", buf_pool->n_pend_reads);
|
||||||
|
|
||||||
|
@ -1819,13 +1820,20 @@ buf_print_io(
|
||||||
mutex_enter(&(buf_pool->mutex));
|
mutex_enter(&(buf_pool->mutex));
|
||||||
|
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"Free list length %lu \n", UT_LIST_GET_LEN(buf_pool->free));
|
"Buffer pool size %lu\n", size);
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"LRU list length %lu \n", UT_LIST_GET_LEN(buf_pool->LRU));
|
"Free buffers %lu\n", UT_LIST_GET_LEN(buf_pool->free));
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"Flush list length %lu \n",
|
"Database pages %lu\n", UT_LIST_GET_LEN(buf_pool->LRU));
|
||||||
|
/*
|
||||||
|
buf += sprintf(buf,
|
||||||
|
"Lock heap buffers %lu\n", buf_pool->n_lock_heap_pages);
|
||||||
|
buf += sprintf(buf,
|
||||||
|
"Hash index buffers %lu\n", buf_pool->n_adaptive_hash_pages);
|
||||||
|
*/
|
||||||
|
buf += sprintf(buf,
|
||||||
|
"Modified db pages %lu\n",
|
||||||
UT_LIST_GET_LEN(buf_pool->flush_list));
|
UT_LIST_GET_LEN(buf_pool->flush_list));
|
||||||
buf += sprintf(buf, "Buffer pool size %lu\n", size);
|
|
||||||
|
|
||||||
buf += sprintf(buf, "Pending reads %lu \n", buf_pool->n_pend_reads);
|
buf += sprintf(buf, "Pending reads %lu \n", buf_pool->n_pend_reads);
|
||||||
|
|
||||||
|
@ -1836,8 +1844,8 @@ buf_print_io(
|
||||||
buf_pool->n_flush[BUF_FLUSH_SINGLE_PAGE]);
|
buf_pool->n_flush[BUF_FLUSH_SINGLE_PAGE]);
|
||||||
|
|
||||||
current_time = time(NULL);
|
current_time = time(NULL);
|
||||||
time_elapsed = difftime(current_time, buf_pool->last_printout_time);
|
time_elapsed = 0.001 + difftime(current_time,
|
||||||
|
buf_pool->last_printout_time);
|
||||||
buf_pool->last_printout_time = current_time;
|
buf_pool->last_printout_time = current_time;
|
||||||
|
|
||||||
buf += sprintf(buf, "Pages read %lu, created %lu, written %lu\n",
|
buf += sprintf(buf, "Pages read %lu, created %lu, written %lu\n",
|
||||||
|
@ -1870,6 +1878,20 @@ buf_print_io(
|
||||||
mutex_exit(&(buf_pool->mutex));
|
mutex_exit(&(buf_pool->mutex));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**************************************************************************
|
||||||
|
Refreshes the statistics used to print per-second averages. */
|
||||||
|
|
||||||
|
void
|
||||||
|
buf_refresh_io_stats(void)
|
||||||
|
/*======================*/
|
||||||
|
{
|
||||||
|
buf_pool->last_printout_time = time(NULL);
|
||||||
|
buf_pool->n_page_gets_old = buf_pool->n_page_gets;
|
||||||
|
buf_pool->n_pages_read_old = buf_pool->n_pages_read;
|
||||||
|
buf_pool->n_pages_created_old = buf_pool->n_pages_created;
|
||||||
|
buf_pool->n_pages_written_old = buf_pool->n_pages_written;
|
||||||
|
}
|
||||||
|
|
||||||
/*************************************************************************
|
/*************************************************************************
|
||||||
Checks that all file pages in the buffer are in a replaceable state. */
|
Checks that all file pages in the buffer are in a replaceable state. */
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,7 @@ Created 11/5/1995 Heikki Tuuri
|
||||||
#include "buf0rea.h"
|
#include "buf0rea.h"
|
||||||
#include "btr0sea.h"
|
#include "btr0sea.h"
|
||||||
#include "os0file.h"
|
#include "os0file.h"
|
||||||
|
#include "log0recv.h"
|
||||||
|
|
||||||
/* The number of blocks from the LRU_old pointer onward, including the block
|
/* The number of blocks from the LRU_old pointer onward, including the block
|
||||||
pointed to, must be 3/8 of the whole LRU list length, except that the
|
pointed to, must be 3/8 of the whole LRU list length, except that the
|
||||||
|
@ -205,6 +206,44 @@ buf_LRU_get_free_block(void)
|
||||||
loop:
|
loop:
|
||||||
mutex_enter(&(buf_pool->mutex));
|
mutex_enter(&(buf_pool->mutex));
|
||||||
|
|
||||||
|
if (!recv_recovery_on && UT_LIST_GET_LEN(buf_pool->free)
|
||||||
|
+ UT_LIST_GET_LEN(buf_pool->LRU) < buf_pool->max_size / 10) {
|
||||||
|
ut_print_timestamp(stderr);
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
" InnoDB: ERROR: over 9 / 10 of the buffer pool is occupied by\n"
|
||||||
|
"InnoDB: lock heaps or the adaptive hash index!\n"
|
||||||
|
"InnoDB: We intentionally generate a seg fault to print a stack trace\n"
|
||||||
|
"InnoDB: on Linux!\n");
|
||||||
|
|
||||||
|
ut_a(0);
|
||||||
|
|
||||||
|
} else if (!recv_recovery_on && UT_LIST_GET_LEN(buf_pool->free)
|
||||||
|
+ UT_LIST_GET_LEN(buf_pool->LRU) < buf_pool->max_size / 5) {
|
||||||
|
|
||||||
|
/* Over 80 % of the buffer pool is occupied by lock heaps
|
||||||
|
or the adaptive hash index. This may be a memory leak! */
|
||||||
|
|
||||||
|
ut_print_timestamp(stderr);
|
||||||
|
fprintf(stderr,
|
||||||
|
" InnoDB: WARNING: over 4 / 5 of the buffer pool is occupied by\n"
|
||||||
|
"InnoDB: lock heaps or the adaptive hash index! Check that your\n"
|
||||||
|
"InnoDB: transactions do not set too many row locks. Starting InnoDB\n"
|
||||||
|
"InnoDB: Monitor to print diagnostics, including lock heap and hash index\n"
|
||||||
|
"InnoDB: sizes.\n");
|
||||||
|
|
||||||
|
srv_print_innodb_monitor = TRUE;
|
||||||
|
|
||||||
|
} else if (!recv_recovery_on && UT_LIST_GET_LEN(buf_pool->free)
|
||||||
|
+ UT_LIST_GET_LEN(buf_pool->LRU) < buf_pool->max_size / 4) {
|
||||||
|
|
||||||
|
/* Switch off the InnoDB Monitor; this is a simple way
|
||||||
|
to stop the monitor if the situation becomes less urgent,
|
||||||
|
but may also surprise users! */
|
||||||
|
|
||||||
|
srv_print_innodb_monitor = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
if (buf_pool->LRU_flush_ended > 0) {
|
if (buf_pool->LRU_flush_ended > 0) {
|
||||||
mutex_exit(&(buf_pool->mutex));
|
mutex_exit(&(buf_pool->mutex));
|
||||||
|
|
||||||
|
|
|
@ -86,8 +86,10 @@ else
|
||||||
fi
|
fi
|
||||||
|
|
||||||
case "$target_os" in
|
case "$target_os" in
|
||||||
|
hpux10*)
|
||||||
|
CFLAGS="$CFLAGS -DUNIV_MUST_NOT_INLINE -DUNIV_HPUX -DUNIV_HPUX10";;
|
||||||
hp*)
|
hp*)
|
||||||
CFLAGS="$CFLAGS -DUNIV_MUST_NOT_INLINE";;
|
CFLAGS="$CFLAGS -DUNIV_MUST_NOT_INLINE -DUNIV_HPUX";;
|
||||||
irix*)
|
irix*)
|
||||||
CFLAGS="$CFLAGS -DUNIV_MUST_NOT_INLINE";;
|
CFLAGS="$CFLAGS -DUNIV_MUST_NOT_INLINE";;
|
||||||
osf*)
|
osf*)
|
||||||
|
|
|
@ -261,7 +261,7 @@ dict_table_get_index_noninline(
|
||||||
{
|
{
|
||||||
return(dict_table_get_index(table, name));
|
return(dict_table_get_index(table, name));
|
||||||
}
|
}
|
||||||
|
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
Initializes the autoinc counter. It is not an error to initialize an already
|
Initializes the autoinc counter. It is not an error to initialize an already
|
||||||
initialized counter. */
|
initialized counter. */
|
||||||
|
@ -270,7 +270,7 @@ void
|
||||||
dict_table_autoinc_initialize(
|
dict_table_autoinc_initialize(
|
||||||
/*==========================*/
|
/*==========================*/
|
||||||
dict_table_t* table, /* in: table */
|
dict_table_t* table, /* in: table */
|
||||||
ib_longlong value) /* in: value which was assigned to a row */
|
ib_longlong value) /* in: next value to assign to a row */
|
||||||
{
|
{
|
||||||
mutex_enter(&(table->autoinc_mutex));
|
mutex_enter(&(table->autoinc_mutex));
|
||||||
|
|
||||||
|
@ -281,8 +281,8 @@ dict_table_autoinc_initialize(
|
||||||
}
|
}
|
||||||
|
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
Gets the next autoinc value, 0 if not yet initialized. If initialized,
|
Gets the next autoinc value (== autoinc counter value), 0 if not yet
|
||||||
increments the counter by 1. */
|
initialized. If initialized, increments the counter by 1. */
|
||||||
|
|
||||||
ib_longlong
|
ib_longlong
|
||||||
dict_table_autoinc_get(
|
dict_table_autoinc_get(
|
||||||
|
@ -298,8 +298,8 @@ dict_table_autoinc_get(
|
||||||
|
|
||||||
value = 0;
|
value = 0;
|
||||||
} else {
|
} else {
|
||||||
table->autoinc = table->autoinc + 1;
|
|
||||||
value = table->autoinc;
|
value = table->autoinc;
|
||||||
|
table->autoinc = table->autoinc + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_exit(&(table->autoinc_mutex));
|
mutex_exit(&(table->autoinc_mutex));
|
||||||
|
@ -334,20 +334,43 @@ dict_table_autoinc_read(
|
||||||
}
|
}
|
||||||
|
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
Updates the autoinc counter if the value supplied is bigger than the
|
Peeks the autoinc counter value, 0 if not yet initialized. Does not
|
||||||
|
increment the counter. The read not protected by any mutex! */
|
||||||
|
|
||||||
|
ib_longlong
|
||||||
|
dict_table_autoinc_peek(
|
||||||
|
/*====================*/
|
||||||
|
/* out: value of the counter */
|
||||||
|
dict_table_t* table) /* in: table */
|
||||||
|
{
|
||||||
|
ib_longlong value;
|
||||||
|
|
||||||
|
if (!table->autoinc_inited) {
|
||||||
|
|
||||||
|
value = 0;
|
||||||
|
} else {
|
||||||
|
value = table->autoinc;
|
||||||
|
}
|
||||||
|
|
||||||
|
return(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/************************************************************************
|
||||||
|
Updates the autoinc counter if the value supplied is equal or bigger than the
|
||||||
current value. If not inited, does nothing. */
|
current value. If not inited, does nothing. */
|
||||||
|
|
||||||
void
|
void
|
||||||
dict_table_autoinc_update(
|
dict_table_autoinc_update(
|
||||||
/*======================*/
|
/*======================*/
|
||||||
|
|
||||||
dict_table_t* table, /* in: table */
|
dict_table_t* table, /* in: table */
|
||||||
ib_longlong value) /* in: value which was assigned to a row */
|
ib_longlong value) /* in: value which was assigned to a row */
|
||||||
{
|
{
|
||||||
mutex_enter(&(table->autoinc_mutex));
|
mutex_enter(&(table->autoinc_mutex));
|
||||||
|
|
||||||
if (table->autoinc_inited) {
|
if (table->autoinc_inited) {
|
||||||
if (value > table->autoinc) {
|
if (value >= table->autoinc) {
|
||||||
table->autoinc = value;
|
table->autoinc = value + 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -578,7 +578,7 @@ fil_read_flushed_lsn_and_arch_log_no(
|
||||||
ulint arch_log_no;
|
ulint arch_log_no;
|
||||||
|
|
||||||
buf2 = ut_malloc(2 * UNIV_PAGE_SIZE);
|
buf2 = ut_malloc(2 * UNIV_PAGE_SIZE);
|
||||||
/* Align the memory for a possibel read from a raw device */
|
/* Align the memory for a possible read from a raw device */
|
||||||
buf = ut_align(buf2, UNIV_PAGE_SIZE);
|
buf = ut_align(buf2, UNIV_PAGE_SIZE);
|
||||||
|
|
||||||
os_file_read(data_file, buf, 0, 0, UNIV_PAGE_SIZE);
|
os_file_read(data_file, buf, 0, 0, UNIV_PAGE_SIZE);
|
||||||
|
|
|
@ -933,6 +933,36 @@ fsp_header_get_free_limit(
|
||||||
return(limit);
|
return(limit);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**************************************************************************
|
||||||
|
Gets the size of the tablespace from the tablespace header. If we do not
|
||||||
|
have an auto-extending data file, this should be equal to the size of the
|
||||||
|
data files. If there is an auto-extending data file, this can be smaller. */
|
||||||
|
|
||||||
|
ulint
|
||||||
|
fsp_header_get_tablespace_size(
|
||||||
|
/*===========================*/
|
||||||
|
/* out: size in pages */
|
||||||
|
ulint space) /* in: space id */
|
||||||
|
{
|
||||||
|
fsp_header_t* header;
|
||||||
|
ulint size;
|
||||||
|
mtr_t mtr;
|
||||||
|
|
||||||
|
ut_a(space == 0); /* We have only one log_fsp_current_... variable */
|
||||||
|
|
||||||
|
mtr_start(&mtr);
|
||||||
|
|
||||||
|
mtr_x_lock(fil_space_get_latch(space), &mtr);
|
||||||
|
|
||||||
|
header = fsp_get_space_header(space, &mtr);
|
||||||
|
|
||||||
|
size = mtr_read_ulint(header + FSP_SIZE, MLOG_4BYTES, &mtr);
|
||||||
|
|
||||||
|
mtr_commit(&mtr);
|
||||||
|
|
||||||
|
return(size);
|
||||||
|
}
|
||||||
|
|
||||||
/***************************************************************************
|
/***************************************************************************
|
||||||
Tries to extend the last data file file if it is defined as auto-extending. */
|
Tries to extend the last data file file if it is defined as auto-extending. */
|
||||||
static
|
static
|
||||||
|
@ -2629,7 +2659,7 @@ fseg_free_page_low(
|
||||||
"InnoDB: Dump of the tablespace extent descriptor: %s\n", errbuf);
|
"InnoDB: Dump of the tablespace extent descriptor: %s\n", errbuf);
|
||||||
|
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
"InnoDB: Serious error! InnoDB is trying to free page %lu\n",
|
"InnoDB: Serious error! InnoDB is trying to free page %lu\n"
|
||||||
"InnoDB: though it is already marked as free in the tablespace!\n"
|
"InnoDB: though it is already marked as free in the tablespace!\n"
|
||||||
"InnoDB: The tablespace free space info is corrupt.\n"
|
"InnoDB: The tablespace free space info is corrupt.\n"
|
||||||
"InnoDB: You may need to dump your InnoDB tables and recreate the whole\n"
|
"InnoDB: You may need to dump your InnoDB tables and recreate the whole\n"
|
||||||
|
|
|
@ -298,6 +298,7 @@ ha_print_info(
|
||||||
ulint cells = 0;
|
ulint cells = 0;
|
||||||
ulint len = 0;
|
ulint len = 0;
|
||||||
ulint max_len = 0;
|
ulint max_len = 0;
|
||||||
|
ulint n_bufs;
|
||||||
ulint i;
|
ulint i;
|
||||||
|
|
||||||
if (buf_end - buf < 200) {
|
if (buf_end - buf < 200) {
|
||||||
|
@ -335,6 +336,20 @@ ha_print_info(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
buf += sprintf(buf, "Hash table size %lu, used cells %lu\n",
|
buf += sprintf(buf,
|
||||||
hash_get_n_cells(table), cells);
|
"Hash table size %lu, used cells %lu", hash_get_n_cells(table), cells);
|
||||||
|
|
||||||
|
if (table->heaps == NULL && table->heap != NULL) {
|
||||||
|
|
||||||
|
/* This calculation is intended for the adaptive hash
|
||||||
|
index: how many buffer frames we have reserved? */
|
||||||
|
|
||||||
|
n_bufs = UT_LIST_GET_LEN(table->heap->base) - 1;
|
||||||
|
|
||||||
|
if (table->heap->free_block) {
|
||||||
|
n_bufs++;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf += sprintf(buf, ", node heap has %lu buffer(s)\n", n_bufs);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -313,14 +313,6 @@ btr_discard_page(
|
||||||
btr_cur_t* cursor, /* in: cursor on the page to discard: not on
|
btr_cur_t* cursor, /* in: cursor on the page to discard: not on
|
||||||
the root page */
|
the root page */
|
||||||
mtr_t* mtr); /* in: mtr */
|
mtr_t* mtr); /* in: mtr */
|
||||||
/************************************************************************
|
|
||||||
Declares the latching order level for the page latch in the debug version. */
|
|
||||||
UNIV_INLINE
|
|
||||||
void
|
|
||||||
btr_declare_page_latch(
|
|
||||||
/*===================*/
|
|
||||||
page_t* page, /* in: page */
|
|
||||||
ibool leaf); /* in: TRUE if a leaf */
|
|
||||||
/********************************************************************
|
/********************************************************************
|
||||||
Parses the redo log record for setting an index record as the predefined
|
Parses the redo log record for setting an index record as the predefined
|
||||||
minimum record. */
|
minimum record. */
|
||||||
|
|
|
@ -463,6 +463,12 @@ buf_print_io(
|
||||||
/*=========*/
|
/*=========*/
|
||||||
char* buf, /* in/out: buffer where to print */
|
char* buf, /* in/out: buffer where to print */
|
||||||
char* buf_end);/* in: buffer end */
|
char* buf_end);/* in: buffer end */
|
||||||
|
/**************************************************************************
|
||||||
|
Refreshes the statistics used to print per-second averages. */
|
||||||
|
|
||||||
|
void
|
||||||
|
buf_refresh_io_stats(void);
|
||||||
|
/*======================*/
|
||||||
/*************************************************************************
|
/*************************************************************************
|
||||||
Checks that all file pages in the buffer are in a replaceable state. */
|
Checks that all file pages in the buffer are in a replaceable state. */
|
||||||
|
|
||||||
|
|
|
@ -211,8 +211,15 @@ buf_block_align(
|
||||||
|
|
||||||
block = buf_pool_get_nth_block(buf_pool, ((ulint)(ptr - frame_zero))
|
block = buf_pool_get_nth_block(buf_pool, ((ulint)(ptr - frame_zero))
|
||||||
>> UNIV_PAGE_SIZE_SHIFT);
|
>> UNIV_PAGE_SIZE_SHIFT);
|
||||||
ut_a(block >= buf_pool->blocks);
|
if (block < buf_pool->blocks
|
||||||
ut_a(block < buf_pool->blocks + buf_pool->max_size);
|
|| block >= buf_pool->blocks + buf_pool->max_size) {
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: trying to access a stray pointer %lx\n"
|
||||||
|
"InnoDB: buf pool start is at %lx, number of pages %lu\n", (ulint)ptr,
|
||||||
|
(ulint)frame_zero, buf_pool->max_size);
|
||||||
|
ut_a(0);
|
||||||
|
}
|
||||||
|
|
||||||
return(block);
|
return(block);
|
||||||
}
|
}
|
||||||
|
@ -238,8 +245,15 @@ buf_block_align_low(
|
||||||
|
|
||||||
block = buf_pool_get_nth_block(buf_pool, ((ulint)(ptr - frame_zero))
|
block = buf_pool_get_nth_block(buf_pool, ((ulint)(ptr - frame_zero))
|
||||||
>> UNIV_PAGE_SIZE_SHIFT);
|
>> UNIV_PAGE_SIZE_SHIFT);
|
||||||
ut_a(block >= buf_pool->blocks);
|
if (block < buf_pool->blocks
|
||||||
ut_a(block < buf_pool->blocks + buf_pool->max_size);
|
|| block >= buf_pool->blocks + buf_pool->max_size) {
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: trying to access a stray pointer %lx\n"
|
||||||
|
"InnoDB: buf pool start is at %lx, number of pages %lu\n", (ulint)ptr,
|
||||||
|
(ulint)frame_zero, buf_pool->max_size);
|
||||||
|
ut_a(0);
|
||||||
|
}
|
||||||
|
|
||||||
return(block);
|
return(block);
|
||||||
}
|
}
|
||||||
|
@ -259,10 +273,17 @@ buf_frame_align(
|
||||||
|
|
||||||
frame = ut_align_down(ptr, UNIV_PAGE_SIZE);
|
frame = ut_align_down(ptr, UNIV_PAGE_SIZE);
|
||||||
|
|
||||||
ut_a((ulint)frame
|
if (((ulint)frame
|
||||||
>= (ulint)(buf_pool_get_nth_block(buf_pool, 0)->frame));
|
< (ulint)(buf_pool->frame_zero))
|
||||||
ut_a((ulint)frame <= (ulint)(buf_pool_get_nth_block(buf_pool,
|
|| ((ulint)frame > (ulint)(buf_pool_get_nth_block(buf_pool,
|
||||||
buf_pool->max_size - 1)->frame));
|
buf_pool->max_size - 1)->frame))) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: trying to access a stray pointer %lx\n"
|
||||||
|
"InnoDB: buf pool start is at %lx, number of pages %lu\n", (ulint)ptr,
|
||||||
|
(ulint)(buf_pool->frame_zero), buf_pool->max_size);
|
||||||
|
ut_a(0);
|
||||||
|
}
|
||||||
|
|
||||||
return(frame);
|
return(frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -96,17 +96,17 @@ dict_col_get_clust_pos(
|
||||||
/*===================*/
|
/*===================*/
|
||||||
dict_col_t* col);
|
dict_col_t* col);
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
Initializes the autoinc counter. It is not an error to initialize already
|
Initializes the autoinc counter. It is not an error to initialize an already
|
||||||
initialized counter. */
|
initialized counter. */
|
||||||
|
|
||||||
void
|
void
|
||||||
dict_table_autoinc_initialize(
|
dict_table_autoinc_initialize(
|
||||||
/*==========================*/
|
/*==========================*/
|
||||||
dict_table_t* table, /* in: table */
|
dict_table_t* table, /* in: table */
|
||||||
ib_longlong value); /* in: value which was assigned to a row */
|
ib_longlong value); /* in: next value to assign to a row */
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
Gets the next autoinc value, 0 if not yet initialized. If initialized,
|
Gets the next autoinc value (== autoinc counter value), 0 if not yet
|
||||||
increments the counter by 1. */
|
initialized. If initialized, increments the counter by 1. */
|
||||||
|
|
||||||
ib_longlong
|
ib_longlong
|
||||||
dict_table_autoinc_get(
|
dict_table_autoinc_get(
|
||||||
|
@ -123,12 +123,22 @@ dict_table_autoinc_read(
|
||||||
/* out: value of the counter */
|
/* out: value of the counter */
|
||||||
dict_table_t* table); /* in: table */
|
dict_table_t* table); /* in: table */
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
Updates the autoinc counter if the value supplied is bigger than the
|
Peeks the autoinc counter value, 0 if not yet initialized. Does not
|
||||||
|
increment the counter. The read not protected by any mutex! */
|
||||||
|
|
||||||
|
ib_longlong
|
||||||
|
dict_table_autoinc_peek(
|
||||||
|
/*====================*/
|
||||||
|
/* out: value of the counter */
|
||||||
|
dict_table_t* table); /* in: table */
|
||||||
|
/************************************************************************
|
||||||
|
Updates the autoinc counter if the value supplied is equal or bigger than the
|
||||||
current value. If not inited, does nothing. */
|
current value. If not inited, does nothing. */
|
||||||
|
|
||||||
void
|
void
|
||||||
dict_table_autoinc_update(
|
dict_table_autoinc_update(
|
||||||
/*======================*/
|
/*======================*/
|
||||||
|
|
||||||
dict_table_t* table, /* in: table */
|
dict_table_t* table, /* in: table */
|
||||||
ib_longlong value); /* in: value which was assigned to a row */
|
ib_longlong value); /* in: value which was assigned to a row */
|
||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
|
|
|
@ -388,8 +388,8 @@ struct dict_table_struct{
|
||||||
/* TRUE if the autoinc counter has been
|
/* TRUE if the autoinc counter has been
|
||||||
inited; MySQL gets the init value by executing
|
inited; MySQL gets the init value by executing
|
||||||
SELECT MAX(auto inc column) */
|
SELECT MAX(auto inc column) */
|
||||||
ib_longlong autoinc;/* autoinc counter value already given to
|
ib_longlong autoinc;/* autoinc counter value to give to the
|
||||||
a row */
|
next inserted row */
|
||||||
ulint magic_n;/* magic number */
|
ulint magic_n;/* magic number */
|
||||||
};
|
};
|
||||||
#define DICT_TABLE_MAGIC_N 76333786
|
#define DICT_TABLE_MAGIC_N 76333786
|
||||||
|
|
|
@ -17,7 +17,8 @@ typedef struct dyn_block_struct dyn_block_t;
|
||||||
typedef dyn_block_t dyn_array_t;
|
typedef dyn_block_t dyn_array_t;
|
||||||
|
|
||||||
|
|
||||||
/* This must be > MLOG_BUF_MARGIN + 30 */
|
/* This is the initial 'payload' size of a dynamic array;
|
||||||
|
this must be > MLOG_BUF_MARGIN + 30! */
|
||||||
#define DYN_ARRAY_DATA_SIZE 512
|
#define DYN_ARRAY_DATA_SIZE 512
|
||||||
|
|
||||||
/*************************************************************************
|
/*************************************************************************
|
||||||
|
@ -123,14 +124,6 @@ dyn_block_get_data(
|
||||||
/*===============*/
|
/*===============*/
|
||||||
/* out: pointer to data */
|
/* out: pointer to data */
|
||||||
dyn_block_t* block); /* in: dyn array block */
|
dyn_block_t* block); /* in: dyn array block */
|
||||||
/************************************************************************
|
|
||||||
Gets the next block in a dyn array. */
|
|
||||||
UNIV_INLINE
|
|
||||||
dyn_block_t*
|
|
||||||
dyn_block_get_next(
|
|
||||||
/*===============*/
|
|
||||||
/* out: pointer to next, NULL if end of list */
|
|
||||||
dyn_block_t* block); /* in: dyn array block */
|
|
||||||
/************************************************************
|
/************************************************************
|
||||||
Pushes n bytes to a dyn array. */
|
Pushes n bytes to a dyn array. */
|
||||||
UNIV_INLINE
|
UNIV_INLINE
|
||||||
|
|
|
@ -57,6 +57,16 @@ fsp_header_get_free_limit(
|
||||||
/* out: free limit in megabytes */
|
/* out: free limit in megabytes */
|
||||||
ulint space); /* in: space id */
|
ulint space); /* in: space id */
|
||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
|
Gets the size of the tablespace from the tablespace header. If we do not
|
||||||
|
have an auto-extending data file, this should be equal to the size of the
|
||||||
|
data files. If there is an auto-extending data file, this can be smaller. */
|
||||||
|
|
||||||
|
ulint
|
||||||
|
fsp_header_get_tablespace_size(
|
||||||
|
/*===========================*/
|
||||||
|
/* out: size in pages */
|
||||||
|
ulint space); /* in: space id */
|
||||||
|
/**************************************************************************
|
||||||
Initializes the space header of a new created space. */
|
Initializes the space header of a new created space. */
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -131,6 +131,14 @@ ha_print_info(
|
||||||
char* buf_end,/* in: buffer end */
|
char* buf_end,/* in: buffer end */
|
||||||
hash_table_t* table); /* in: hash table */
|
hash_table_t* table); /* in: hash table */
|
||||||
|
|
||||||
|
/* The hash table external chain node */
|
||||||
|
|
||||||
|
typedef struct ha_node_struct ha_node_t;
|
||||||
|
struct ha_node_struct {
|
||||||
|
ha_node_t* next; /* next chain node or NULL if none */
|
||||||
|
void* data; /* pointer to the data */
|
||||||
|
ulint fold; /* fold value for the data */
|
||||||
|
};
|
||||||
|
|
||||||
#ifndef UNIV_NONINL
|
#ifndef UNIV_NONINL
|
||||||
#include "ha0ha.ic"
|
#include "ha0ha.ic"
|
||||||
|
|
|
@ -9,16 +9,6 @@ Created 8/18/1994 Heikki Tuuri
|
||||||
#include "ut0rnd.h"
|
#include "ut0rnd.h"
|
||||||
#include "mem0mem.h"
|
#include "mem0mem.h"
|
||||||
|
|
||||||
/* The hash table external chain node */
|
|
||||||
|
|
||||||
typedef struct ha_node_struct ha_node_t;
|
|
||||||
|
|
||||||
struct ha_node_struct {
|
|
||||||
ha_node_t* next; /* next chain node or NULL if none */
|
|
||||||
void* data; /* pointer to the data */
|
|
||||||
ulint fold; /* fold value for the data */
|
|
||||||
};
|
|
||||||
|
|
||||||
/***************************************************************
|
/***************************************************************
|
||||||
Deletes a hash node. */
|
Deletes a hash node. */
|
||||||
|
|
||||||
|
|
|
@ -431,15 +431,30 @@ log_block_set_data_len(
|
||||||
byte* log_block, /* in: log block */
|
byte* log_block, /* in: log block */
|
||||||
ulint len); /* in: data length */
|
ulint len); /* in: data length */
|
||||||
/****************************************************************
|
/****************************************************************
|
||||||
Gets a log block number stored in the trailer. */
|
Calculates the checksum for a log block. */
|
||||||
UNIV_INLINE
|
UNIV_INLINE
|
||||||
ulint
|
ulint
|
||||||
log_block_get_trl_no(
|
log_block_calc_checksum(
|
||||||
/*=================*/
|
/*====================*/
|
||||||
/* out: log block number stored in the block
|
/* out: checksum */
|
||||||
trailer */
|
byte* block); /* in: log block */
|
||||||
|
/****************************************************************
|
||||||
|
Gets a log block checksum field value. */
|
||||||
|
UNIV_INLINE
|
||||||
|
ulint
|
||||||
|
log_block_get_checksum(
|
||||||
|
/*===================*/
|
||||||
|
/* out: checksum */
|
||||||
byte* log_block); /* in: log block */
|
byte* log_block); /* in: log block */
|
||||||
/****************************************************************
|
/****************************************************************
|
||||||
|
Sets a log block checksum field value. */
|
||||||
|
UNIV_INLINE
|
||||||
|
void
|
||||||
|
log_block_set_checksum(
|
||||||
|
/*===================*/
|
||||||
|
byte* log_block, /* in: log block */
|
||||||
|
ulint checksum); /* in: checksum */
|
||||||
|
/****************************************************************
|
||||||
Gets a log block first mtr log record group offset. */
|
Gets a log block first mtr log record group offset. */
|
||||||
UNIV_INLINE
|
UNIV_INLINE
|
||||||
ulint
|
ulint
|
||||||
|
@ -497,6 +512,12 @@ log_print(
|
||||||
/*======*/
|
/*======*/
|
||||||
char* buf, /* in/out: buffer where to print */
|
char* buf, /* in/out: buffer where to print */
|
||||||
char* buf_end);/* in: buffer end */
|
char* buf_end);/* in: buffer end */
|
||||||
|
/**************************************************************************
|
||||||
|
Refreshes the statistics used to print per-second averages. */
|
||||||
|
|
||||||
|
void
|
||||||
|
log_refresh_stats(void);
|
||||||
|
/*===================*/
|
||||||
|
|
||||||
extern log_t* log_sys;
|
extern log_t* log_sys;
|
||||||
|
|
||||||
|
@ -544,10 +565,11 @@ extern log_t* log_sys;
|
||||||
bytes */
|
bytes */
|
||||||
|
|
||||||
/* Offsets of a log block trailer from the end of the block */
|
/* Offsets of a log block trailer from the end of the block */
|
||||||
#define LOG_BLOCK_TRL_CHECKSUM 4 /* 1 byte checksum of the log block
|
#define LOG_BLOCK_CHECKSUM 4 /* 4 byte checksum of the log block
|
||||||
contents */
|
contents; in InnoDB versions
|
||||||
#define LOG_BLOCK_TRL_NO 3 /* 3 lowest bytes of the log block
|
< 3.23.52 this did not contain the
|
||||||
number */
|
checksum but the same value as
|
||||||
|
.._HDR_NO */
|
||||||
#define LOG_BLOCK_TRL_SIZE 4 /* trailer size in bytes */
|
#define LOG_BLOCK_TRL_SIZE 4 /* trailer size in bytes */
|
||||||
|
|
||||||
/* Offsets for a checkpoint field */
|
/* Offsets for a checkpoint field */
|
||||||
|
|
|
@ -169,33 +169,6 @@ log_block_set_checkpoint_no(
|
||||||
ut_dulint_get_low(no));
|
ut_dulint_get_low(no));
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************
|
|
||||||
Gets a log block number stored in the trailer. */
|
|
||||||
UNIV_INLINE
|
|
||||||
ulint
|
|
||||||
log_block_get_trl_no(
|
|
||||||
/*=================*/
|
|
||||||
/* out: log block number stored in the block
|
|
||||||
trailer */
|
|
||||||
byte* log_block) /* in: log block */
|
|
||||||
{
|
|
||||||
return(mach_read_from_3(log_block + OS_FILE_LOG_BLOCK_SIZE
|
|
||||||
- LOG_BLOCK_TRL_NO));
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************
|
|
||||||
Sets the log block number stored in the trailer. */
|
|
||||||
UNIV_INLINE
|
|
||||||
void
|
|
||||||
log_block_set_trl_no(
|
|
||||||
/*=================*/
|
|
||||||
byte* log_block, /* in: log block */
|
|
||||||
ulint n) /* in: log block number */
|
|
||||||
{
|
|
||||||
mach_write_to_3(log_block + OS_FILE_LOG_BLOCK_SIZE - LOG_BLOCK_TRL_NO,
|
|
||||||
n & 0xFFFFFF);
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************
|
/****************************************************************
|
||||||
Converts a lsn to a log block number. */
|
Converts a lsn to a log block number. */
|
||||||
UNIV_INLINE
|
UNIV_INLINE
|
||||||
|
@ -216,6 +189,61 @@ log_block_convert_lsn_to_no(
|
||||||
return(no + 1);
|
return(no + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/****************************************************************
|
||||||
|
Calculates the checksum for a log block. */
|
||||||
|
UNIV_INLINE
|
||||||
|
ulint
|
||||||
|
log_block_calc_checksum(
|
||||||
|
/*====================*/
|
||||||
|
/* out: checksum */
|
||||||
|
byte* block) /* in: log block */
|
||||||
|
{
|
||||||
|
ulint sum;
|
||||||
|
ulint sh;
|
||||||
|
ulint i;
|
||||||
|
|
||||||
|
sum = 1;
|
||||||
|
sh = 0;
|
||||||
|
|
||||||
|
for (i = 0; i < OS_FILE_LOG_BLOCK_SIZE - LOG_BLOCK_TRL_SIZE; i++) {
|
||||||
|
sum = sum & 0x7FFFFFFF;
|
||||||
|
sum += (((ulint)(*(block + i))) << sh) + (ulint)(*(block + i));
|
||||||
|
sh++;
|
||||||
|
if (sh > 24) {
|
||||||
|
sh = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return(sum);
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************
|
||||||
|
Gets a log block checksum field value. */
|
||||||
|
UNIV_INLINE
|
||||||
|
ulint
|
||||||
|
log_block_get_checksum(
|
||||||
|
/*===================*/
|
||||||
|
/* out: checksum */
|
||||||
|
byte* log_block) /* in: log block */
|
||||||
|
{
|
||||||
|
return(mach_read_from_4(log_block + OS_FILE_LOG_BLOCK_SIZE
|
||||||
|
- LOG_BLOCK_CHECKSUM));
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************
|
||||||
|
Sets a log block checksum field value. */
|
||||||
|
UNIV_INLINE
|
||||||
|
void
|
||||||
|
log_block_set_checksum(
|
||||||
|
/*===================*/
|
||||||
|
byte* log_block, /* in: log block */
|
||||||
|
ulint checksum) /* in: checksum */
|
||||||
|
{
|
||||||
|
mach_write_to_4(log_block + OS_FILE_LOG_BLOCK_SIZE
|
||||||
|
- LOG_BLOCK_CHECKSUM,
|
||||||
|
checksum);
|
||||||
|
}
|
||||||
|
|
||||||
/****************************************************************
|
/****************************************************************
|
||||||
Initializes a log block in the log buffer. */
|
Initializes a log block in the log buffer. */
|
||||||
UNIV_INLINE
|
UNIV_INLINE
|
||||||
|
@ -232,7 +260,6 @@ log_block_init(
|
||||||
no = log_block_convert_lsn_to_no(lsn);
|
no = log_block_convert_lsn_to_no(lsn);
|
||||||
|
|
||||||
log_block_set_hdr_no(log_block, no);
|
log_block_set_hdr_no(log_block, no);
|
||||||
log_block_set_trl_no(log_block, no);
|
|
||||||
|
|
||||||
log_block_set_data_len(log_block, LOG_BLOCK_HDR_SIZE);
|
log_block_set_data_len(log_block, LOG_BLOCK_HDR_SIZE);
|
||||||
log_block_set_first_rec_group(log_block, 0);
|
log_block_set_first_rec_group(log_block, 0);
|
||||||
|
@ -256,7 +283,7 @@ log_block_init_in_old_format(
|
||||||
|
|
||||||
log_block_set_hdr_no(log_block, no);
|
log_block_set_hdr_no(log_block, no);
|
||||||
mach_write_to_4(log_block + OS_FILE_LOG_BLOCK_SIZE
|
mach_write_to_4(log_block + OS_FILE_LOG_BLOCK_SIZE
|
||||||
- LOG_BLOCK_TRL_NO - 1, no);
|
- LOG_BLOCK_CHECKSUM, no);
|
||||||
log_block_set_data_len(log_block, LOG_BLOCK_HDR_SIZE);
|
log_block_set_data_len(log_block, LOG_BLOCK_HDR_SIZE);
|
||||||
log_block_set_first_rec_group(log_block, 0);
|
log_block_set_first_rec_group(log_block, 0);
|
||||||
}
|
}
|
||||||
|
|
|
@ -313,6 +313,11 @@ struct recv_sys_struct{
|
||||||
this lsn */
|
this lsn */
|
||||||
dulint limit_lsn;/* recovery should be made at most up to this
|
dulint limit_lsn;/* recovery should be made at most up to this
|
||||||
lsn */
|
lsn */
|
||||||
|
ibool found_corrupt_log;
|
||||||
|
/* this is set to TRUE if we during log
|
||||||
|
scan find a corrupt log block, or a corrupt
|
||||||
|
log record, or there is a log parsing
|
||||||
|
buffer overflow */
|
||||||
log_group_t* archive_group;
|
log_group_t* archive_group;
|
||||||
/* in archive recovery: the log group whose
|
/* in archive recovery: the log group whose
|
||||||
archive is read */
|
archive is read */
|
||||||
|
@ -328,6 +333,8 @@ extern ibool recv_recovery_on;
|
||||||
extern ibool recv_no_ibuf_operations;
|
extern ibool recv_no_ibuf_operations;
|
||||||
extern ibool recv_needed_recovery;
|
extern ibool recv_needed_recovery;
|
||||||
|
|
||||||
|
extern ibool recv_is_making_a_backup;
|
||||||
|
|
||||||
/* Size of the parsing buffer; it must accommodate RECV_SCAN_SIZE many
|
/* Size of the parsing buffer; it must accommodate RECV_SCAN_SIZE many
|
||||||
times! */
|
times! */
|
||||||
#define RECV_PARSING_BUF_SIZE (2 * 1024 * 1024)
|
#define RECV_PARSING_BUF_SIZE (2 * 1024 * 1024)
|
||||||
|
|
|
@ -408,6 +408,12 @@ os_aio_print(
|
||||||
char* buf, /* in/out: buffer where to print */
|
char* buf, /* in/out: buffer where to print */
|
||||||
char* buf_end);/* in: buffer end */
|
char* buf_end);/* in: buffer end */
|
||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
|
Refreshes the statistics used to print per-second averages. */
|
||||||
|
|
||||||
|
void
|
||||||
|
os_aio_refresh_stats(void);
|
||||||
|
/*======================*/
|
||||||
|
/**************************************************************************
|
||||||
Checks that all slots in the system have been freed, that is, there are
|
Checks that all slots in the system have been freed, that is, there are
|
||||||
no pending io operations. */
|
no pending io operations. */
|
||||||
|
|
||||||
|
|
|
@ -27,7 +27,21 @@ os_fast_mutex_trylock(
|
||||||
|
|
||||||
return(0);
|
return(0);
|
||||||
#else
|
#else
|
||||||
|
#if defined(UNIV_HOTBACKUP) && defined(UNIV_HPUX10)
|
||||||
|
/* Since the hot backup version is standalone, MySQL does not redefine
|
||||||
|
pthread_mutex_trylock for HP-UX-10.20, and consequently we must invert
|
||||||
|
the return value here */
|
||||||
|
|
||||||
|
return((ulint) (1 - pthread_mutex_trylock(fast_mutex)));
|
||||||
|
#else
|
||||||
|
/* NOTE that the MySQL my_pthread.h redefines pthread_mutex_trylock
|
||||||
|
so that it returns 0 on success. In the operating system
|
||||||
|
libraries, HP-UX-10.20 follows the old Posix 1003.4a Draft 4 and
|
||||||
|
returns 1 on success (but MySQL remaps that to 0), while Linux,
|
||||||
|
FreeBSD, Solaris, AIX, Tru64 Unix, HP-UX-11.0 return 0 on success. */
|
||||||
|
|
||||||
return((ulint) pthread_mutex_trylock(fast_mutex));
|
return((ulint) pthread_mutex_trylock(fast_mutex));
|
||||||
#endif
|
#endif
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -28,12 +28,30 @@ typedef void* os_thread_t;
|
||||||
#else
|
#else
|
||||||
typedef pthread_t os_thread_t;
|
typedef pthread_t os_thread_t;
|
||||||
#endif
|
#endif
|
||||||
typedef unsigned long int os_thread_id_t;
|
|
||||||
|
#define os_thread_id_t os_thread_t
|
||||||
|
|
||||||
/* Define a function pointer type to use in a typecast */
|
/* Define a function pointer type to use in a typecast */
|
||||||
typedef void* (*os_posix_f_t) (void*);
|
typedef void* (*os_posix_f_t) (void*);
|
||||||
|
|
||||||
|
/*******************************************************************
|
||||||
|
Compares two threads or thread ids for equality */
|
||||||
|
|
||||||
|
ibool
|
||||||
|
os_thread_eq(
|
||||||
|
/*=========*/
|
||||||
|
/* out: TRUE if equal */
|
||||||
|
os_thread_t a, /* in: OS thread or thread id */
|
||||||
|
os_thread_t b); /* in: OS thread or thread id */
|
||||||
|
/********************************************************************
|
||||||
|
Converts an OS thread or thread id to a ulint. It is NOT guaranteed that
|
||||||
|
the ulint is unique for the thread though! */
|
||||||
|
|
||||||
|
ulint
|
||||||
|
os_thread_pf(
|
||||||
|
/*=========*/
|
||||||
|
/* out: unsigned long int */
|
||||||
|
os_thread_t a); /* in: thread or thread id */
|
||||||
/********************************************************************
|
/********************************************************************
|
||||||
Creates a new thread of execution. The execution starts from
|
Creates a new thread of execution. The execution starts from
|
||||||
the function given. The start function takes a void* parameter
|
the function given. The start function takes a void* parameter
|
||||||
|
@ -73,14 +91,6 @@ os_thread_t
|
||||||
os_thread_get_curr(void);
|
os_thread_get_curr(void);
|
||||||
/*====================*/
|
/*====================*/
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
Converts a thread id to a ulint. */
|
|
||||||
|
|
||||||
ulint
|
|
||||||
os_thread_conv_id_to_ulint(
|
|
||||||
/*=======================*/
|
|
||||||
/* out: converted to ulint */
|
|
||||||
os_thread_id_t id); /* in: thread id */
|
|
||||||
/*********************************************************************
|
|
||||||
Waits for a thread to terminate. */
|
Waits for a thread to terminate. */
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -85,7 +85,7 @@ extern ibool srv_is_being_shut_down;
|
||||||
/* At a shutdown the value first climbs from 0 to SRV_SHUTDOWN_CLEANUP
|
/* At a shutdown the value first climbs from 0 to SRV_SHUTDOWN_CLEANUP
|
||||||
and then to SRV_SHUTDOWN_LAST_PHASE */
|
and then to SRV_SHUTDOWN_LAST_PHASE */
|
||||||
|
|
||||||
extern ulint srv_shutdown_state;
|
extern ulint srv_shutdown_state;
|
||||||
|
|
||||||
#define SRV_SHUTDOWN_CLEANUP 1
|
#define SRV_SHUTDOWN_CLEANUP 1
|
||||||
#define SRV_SHUTDOWN_LAST_PHASE 2
|
#define SRV_SHUTDOWN_LAST_PHASE 2
|
||||||
|
|
|
@ -312,7 +312,8 @@ rw_lock_x_lock_func_nowait(
|
||||||
&& ((rw_lock_get_writer(lock) == RW_LOCK_NOT_LOCKED)
|
&& ((rw_lock_get_writer(lock) == RW_LOCK_NOT_LOCKED)
|
||||||
|| ((rw_lock_get_writer(lock) == RW_LOCK_EX)
|
|| ((rw_lock_get_writer(lock) == RW_LOCK_EX)
|
||||||
&& (lock->pass == 0)
|
&& (lock->pass == 0)
|
||||||
&& (lock->writer_thread == os_thread_get_curr_id())))) {
|
&& os_thread_eq(lock->writer_thread,
|
||||||
|
os_thread_get_curr_id())))) {
|
||||||
|
|
||||||
rw_lock_set_writer(lock, RW_LOCK_EX);
|
rw_lock_set_writer(lock, RW_LOCK_EX);
|
||||||
lock->writer_thread = os_thread_get_curr_id();
|
lock->writer_thread = os_thread_get_curr_id();
|
||||||
|
|
|
@ -104,6 +104,10 @@ mutex_test_and_set(
|
||||||
ret = os_fast_mutex_trylock(&(mutex->os_fast_mutex));
|
ret = os_fast_mutex_trylock(&(mutex->os_fast_mutex));
|
||||||
|
|
||||||
if (ret == 0) {
|
if (ret == 0) {
|
||||||
|
/* We check that os_fast_mutex_trylock does not leak
|
||||||
|
and allow race conditions */
|
||||||
|
ut_a(mutex->lock_word == 0);
|
||||||
|
|
||||||
mutex->lock_word = 1;
|
mutex->lock_word = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -257,6 +257,15 @@ void
|
||||||
trx_sys_print_mysql_binlog_offset(void);
|
trx_sys_print_mysql_binlog_offset(void);
|
||||||
/*===================================*/
|
/*===================================*/
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
|
Prints to stdout the MySQL binlog info in the system header if the
|
||||||
|
magic number shows it valid. */
|
||||||
|
|
||||||
|
void
|
||||||
|
trx_sys_print_mysql_binlog_offset_from_page(
|
||||||
|
/*========================================*/
|
||||||
|
byte* page); /* in: buffer containing the trx system header page,
|
||||||
|
i.e., page number TRX_SYS_PAGE_NO in the tablespace */
|
||||||
|
/*********************************************************************
|
||||||
Prints to stderr the MySQL master log offset info in the trx system header if
|
Prints to stderr the MySQL master log offset info in the trx system header if
|
||||||
the magic number shows it valid. */
|
the magic number shows it valid. */
|
||||||
|
|
||||||
|
@ -300,11 +309,11 @@ therefore 256; each slot is currently 8 bytes in size */
|
||||||
#define TRX_SYS_MYSQL_LOG_NAME_LEN 512
|
#define TRX_SYS_MYSQL_LOG_NAME_LEN 512
|
||||||
#define TRX_SYS_MYSQL_LOG_MAGIC_N 873422344
|
#define TRX_SYS_MYSQL_LOG_MAGIC_N 873422344
|
||||||
|
|
||||||
/* The offset of the MySQL replication info on the trx system header page;
|
/* The offset of the MySQL replication info in the trx system header;
|
||||||
this contains the same fields as TRX_SYS_MYSQL_LOG_INFO below */
|
this contains the same fields as TRX_SYS_MYSQL_LOG_INFO below */
|
||||||
#define TRX_SYS_MYSQL_MASTER_LOG_INFO (UNIV_PAGE_SIZE - 2000)
|
#define TRX_SYS_MYSQL_MASTER_LOG_INFO (UNIV_PAGE_SIZE - 2000)
|
||||||
|
|
||||||
/* The offset of the MySQL binlog offset info on the trx system header page */
|
/* The offset of the MySQL binlog offset info in the trx system header */
|
||||||
#define TRX_SYS_MYSQL_LOG_INFO (UNIV_PAGE_SIZE - 1000)
|
#define TRX_SYS_MYSQL_LOG_INFO (UNIV_PAGE_SIZE - 1000)
|
||||||
#define TRX_SYS_MYSQL_LOG_MAGIC_N_FLD 0 /* magic number which shows
|
#define TRX_SYS_MYSQL_LOG_MAGIC_N_FLD 0 /* magic number which shows
|
||||||
if we have valid data in the
|
if we have valid data in the
|
||||||
|
|
|
@ -309,6 +309,9 @@ struct trx_struct{
|
||||||
of view of concurrency control:
|
of view of concurrency control:
|
||||||
TRX_ACTIVE, TRX_COMMITTED_IN_MEMORY,
|
TRX_ACTIVE, TRX_COMMITTED_IN_MEMORY,
|
||||||
... */
|
... */
|
||||||
|
time_t start_time; /* time the trx object was created
|
||||||
|
or the state last time became
|
||||||
|
TRX_ACTIVE */
|
||||||
ibool check_foreigns; /* normally TRUE, but if the user
|
ibool check_foreigns; /* normally TRUE, but if the user
|
||||||
wants to suppress foreign key checks,
|
wants to suppress foreign key checks,
|
||||||
(in table imports, for example) we
|
(in table imports, for example) we
|
||||||
|
@ -468,6 +471,7 @@ struct trx_struct{
|
||||||
TRX_QUE_LOCK_WAIT, this points to
|
TRX_QUE_LOCK_WAIT, this points to
|
||||||
the lock request, otherwise this is
|
the lock request, otherwise this is
|
||||||
NULL */
|
NULL */
|
||||||
|
time_t wait_started; /* lock wait started at this time */
|
||||||
UT_LIST_BASE_NODE_T(que_thr_t)
|
UT_LIST_BASE_NODE_T(que_thr_t)
|
||||||
wait_thrs; /* query threads belonging to this
|
wait_thrs; /* query threads belonging to this
|
||||||
trx that are in the QUE_THR_LOCK_WAIT
|
trx that are in the QUE_THR_LOCK_WAIT
|
||||||
|
|
|
@ -80,8 +80,8 @@ memory is read outside the allocated blocks. */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define UNIV_DEBUG
|
#define UNIV_DEBUG
|
||||||
#define UNIV_MEM_DEBUG
|
|
||||||
#define UNIV_SYNC_DEBUG
|
#define UNIV_SYNC_DEBUG
|
||||||
|
#define UNIV_MEM_DEBUG
|
||||||
|
|
||||||
#define UNIV_IBUF_DEBUG
|
#define UNIV_IBUF_DEBUG
|
||||||
#define UNIV_SEARCH_DEBUG
|
#define UNIV_SEARCH_DEBUG
|
||||||
|
@ -116,7 +116,7 @@ memory is read outside the allocated blocks. */
|
||||||
#define UNIV_INLINE extern inline
|
#define UNIV_INLINE extern inline
|
||||||
#else
|
#else
|
||||||
/* extern inline doesn't work with gcc 3.0.2 */
|
/* extern inline doesn't work with gcc 3.0.2 */
|
||||||
#define UNIV_INLINE static inline
|
#define UNIV_INLINE static inline
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -204,8 +204,12 @@ headers may define 'bool' differently. Do not assume that 'bool' is a ulint! */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* The following number as the length of a logical field means that the field
|
/* The following number as the length of a logical field means that the field
|
||||||
has the SQL NULL as its value. */
|
has the SQL NULL as its value. NOTE that because we assume that the length
|
||||||
#define UNIV_SQL_NULL ULINT_UNDEFINED
|
of a field is a 32-bit integer when we store it, for example, to an undo log
|
||||||
|
on disk, we must have also this number fit in 32 bits, also in 64-bit
|
||||||
|
computers! */
|
||||||
|
|
||||||
|
#define UNIV_SQL_NULL ULINT32_UNDEFINED
|
||||||
|
|
||||||
/* Lengths which are not UNIV_SQL_NULL, but bigger than the following
|
/* Lengths which are not UNIV_SQL_NULL, but bigger than the following
|
||||||
number indicate that a field contains a reference to an externally
|
number indicate that a field contains a reference to an externally
|
||||||
|
|
|
@ -26,9 +26,11 @@ extern ulint* ut_dbg_null_ptr;
|
||||||
ulint dbg_i;\
|
ulint dbg_i;\
|
||||||
\
|
\
|
||||||
if (!((ulint)(EXPR) + ut_dbg_zero)) {\
|
if (!((ulint)(EXPR) + ut_dbg_zero)) {\
|
||||||
|
ut_print_timestamp(stderr);\
|
||||||
fprintf(stderr,\
|
fprintf(stderr,\
|
||||||
"InnoDB: Assertion failure in thread %lu in file %s line %lu\n",\
|
" InnoDB: Assertion failure in thread %lu in file %s line %lu\n",\
|
||||||
os_thread_get_curr_id(), IB__FILE__, (ulint)__LINE__);\
|
os_thread_pf(os_thread_get_curr_id()), IB__FILE__,\
|
||||||
|
(ulint)__LINE__);\
|
||||||
fprintf(stderr,\
|
fprintf(stderr,\
|
||||||
"InnoDB: We intentionally generate a memory trap.\n");\
|
"InnoDB: We intentionally generate a memory trap.\n");\
|
||||||
fprintf(stderr,\
|
fprintf(stderr,\
|
||||||
|
@ -42,16 +44,17 @@ extern ulint* ut_dbg_null_ptr;
|
||||||
if (ut_dbg_stop_threads) {\
|
if (ut_dbg_stop_threads) {\
|
||||||
fprintf(stderr,\
|
fprintf(stderr,\
|
||||||
"InnoDB: Thread %lu stopped in file %s line %lu\n",\
|
"InnoDB: Thread %lu stopped in file %s line %lu\n",\
|
||||||
os_thread_get_curr_id(), IB__FILE__, (ulint)__LINE__);\
|
os_thread_pf(os_thread_get_curr_id()), IB__FILE__, (ulint)__LINE__);\
|
||||||
os_thread_sleep(1000000000);\
|
os_thread_sleep(1000000000);\
|
||||||
}\
|
}\
|
||||||
}
|
}
|
||||||
|
|
||||||
#define ut_error {\
|
#define ut_error {\
|
||||||
ulint dbg_i;\
|
ulint dbg_i;\
|
||||||
|
ut_print_timestamp(stderr);\
|
||||||
fprintf(stderr,\
|
fprintf(stderr,\
|
||||||
"InnoDB: Assertion failure in thread %lu in file %s line %lu\n",\
|
" InnoDB: Assertion failure in thread %lu in file %s line %lu\n",\
|
||||||
os_thread_get_curr_id(), IB__FILE__, (ulint)__LINE__);\
|
os_thread_pf(os_thread_get_curr_id()), IB__FILE__, (ulint)__LINE__);\
|
||||||
fprintf(stderr,\
|
fprintf(stderr,\
|
||||||
"InnoDB: We intentionally generate a memory trap.\n");\
|
"InnoDB: We intentionally generate a memory trap.\n");\
|
||||||
fprintf(stderr,\
|
fprintf(stderr,\
|
||||||
|
|
|
@ -301,6 +301,11 @@ struct lock_struct{
|
||||||
} un_member;
|
} un_member;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* We store info on the latest deadlock error to this buffer. InnoDB
|
||||||
|
Monitor will then fetch it and print */
|
||||||
|
ibool lock_deadlock_found = FALSE;
|
||||||
|
char* lock_latest_err_buf; /* We allocate 5000 bytes for this */
|
||||||
|
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
Checks if a lock request results in a deadlock. */
|
Checks if a lock request results in a deadlock. */
|
||||||
static
|
static
|
||||||
|
@ -576,6 +581,8 @@ lock_sys_create(
|
||||||
lock_sys->rec_hash = hash_create(n_cells);
|
lock_sys->rec_hash = hash_create(n_cells);
|
||||||
|
|
||||||
/* hash_create_mutexes(lock_sys->rec_hash, 2, SYNC_REC_LOCK); */
|
/* hash_create_mutexes(lock_sys->rec_hash, 2, SYNC_REC_LOCK); */
|
||||||
|
|
||||||
|
lock_latest_err_buf = mem_alloc(5000);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*************************************************************************
|
/*************************************************************************
|
||||||
|
@ -1566,6 +1573,7 @@ index->table_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
trx->que_state = TRX_QUE_LOCK_WAIT;
|
trx->que_state = TRX_QUE_LOCK_WAIT;
|
||||||
|
trx->wait_started = time(NULL);
|
||||||
|
|
||||||
ut_a(que_thr_stop(thr));
|
ut_a(que_thr_stop(thr));
|
||||||
|
|
||||||
|
@ -2698,6 +2706,7 @@ lock_deadlock_occurs(
|
||||||
trx_t* mark_trx;
|
trx_t* mark_trx;
|
||||||
ibool ret;
|
ibool ret;
|
||||||
ulint cost = 0;
|
ulint cost = 0;
|
||||||
|
char* err_buf;
|
||||||
|
|
||||||
ut_ad(trx && lock);
|
ut_ad(trx && lock);
|
||||||
ut_ad(mutex_own(&kernel_mutex));
|
ut_ad(mutex_own(&kernel_mutex));
|
||||||
|
@ -2723,6 +2732,29 @@ lock_deadlock_occurs(
|
||||||
index = lock->index;
|
index = lock->index;
|
||||||
table = index->table;
|
table = index->table;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
lock_deadlock_found = TRUE;
|
||||||
|
|
||||||
|
err_buf = lock_latest_err_buf + strlen(lock_latest_err_buf);
|
||||||
|
|
||||||
|
err_buf += sprintf(err_buf,
|
||||||
|
"*** (2) WAITING FOR THIS LOCK TO BE GRANTED:\n");
|
||||||
|
|
||||||
|
ut_a(err_buf <= lock_latest_err_buf + 4000);
|
||||||
|
|
||||||
|
if (lock_get_type(lock) == LOCK_REC) {
|
||||||
|
lock_rec_print(err_buf, lock);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
} else {
|
||||||
|
lock_table_print(err_buf, lock);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
ut_a(err_buf <= lock_latest_err_buf + 4000);
|
||||||
|
|
||||||
|
err_buf += sprintf(err_buf,
|
||||||
|
"*** WE ROLL BACK TRANSACTION (2)\n");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
sess_raise_error_low(trx, DB_DEADLOCK, lock->type_mode, table,
|
sess_raise_error_low(trx, DB_DEADLOCK, lock->type_mode, table,
|
||||||
index, NULL, NULL, NULL);
|
index, NULL, NULL, NULL);
|
||||||
|
@ -2750,6 +2782,7 @@ lock_deadlock_recursive(
|
||||||
lock_t* lock;
|
lock_t* lock;
|
||||||
ulint bit_no = 0; /* remove warning */
|
ulint bit_no = 0; /* remove warning */
|
||||||
trx_t* lock_trx;
|
trx_t* lock_trx;
|
||||||
|
char* err_buf;
|
||||||
|
|
||||||
ut_a(trx && start && wait_lock);
|
ut_a(trx && start && wait_lock);
|
||||||
ut_ad(mutex_own(&kernel_mutex));
|
ut_ad(mutex_own(&kernel_mutex));
|
||||||
|
@ -2801,6 +2834,53 @@ lock_deadlock_recursive(
|
||||||
lock_trx = lock->trx;
|
lock_trx = lock->trx;
|
||||||
|
|
||||||
if (lock_trx == start) {
|
if (lock_trx == start) {
|
||||||
|
err_buf = lock_latest_err_buf;
|
||||||
|
|
||||||
|
ut_sprintf_timestamp(err_buf);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
|
||||||
|
err_buf += sprintf(err_buf,
|
||||||
|
" LATEST DETECTED DEADLOCK:\n"
|
||||||
|
"*** (1) TRANSACTION:\n");
|
||||||
|
|
||||||
|
trx_print(err_buf, wait_lock->trx);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
|
||||||
|
err_buf += sprintf(err_buf,
|
||||||
|
"*** (1) WAITING FOR THIS LOCK TO BE GRANTED:\n");
|
||||||
|
|
||||||
|
ut_a(err_buf <= lock_latest_err_buf + 4000);
|
||||||
|
|
||||||
|
if (lock_get_type(wait_lock) == LOCK_REC) {
|
||||||
|
lock_rec_print(err_buf, wait_lock);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
} else {
|
||||||
|
lock_table_print(err_buf, wait_lock);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
ut_a(err_buf <= lock_latest_err_buf + 4000);
|
||||||
|
err_buf += sprintf(err_buf,
|
||||||
|
"*** (2) TRANSACTION:\n");
|
||||||
|
|
||||||
|
trx_print(err_buf, lock->trx);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
|
||||||
|
err_buf += sprintf(err_buf,
|
||||||
|
"*** (2) HOLDS THE LOCK(S):\n");
|
||||||
|
|
||||||
|
ut_a(err_buf <= lock_latest_err_buf + 4000);
|
||||||
|
|
||||||
|
if (lock_get_type(lock) == LOCK_REC) {
|
||||||
|
lock_rec_print(err_buf, lock);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
} else {
|
||||||
|
lock_table_print(err_buf, lock);
|
||||||
|
err_buf += strlen(err_buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
ut_a(err_buf <= lock_latest_err_buf + 4000);
|
||||||
|
|
||||||
if (lock_print_waits) {
|
if (lock_print_waits) {
|
||||||
printf("Deadlock detected\n");
|
printf("Deadlock detected\n");
|
||||||
}
|
}
|
||||||
|
@ -2962,6 +3042,7 @@ table->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
trx->que_state = TRX_QUE_LOCK_WAIT;
|
trx->que_state = TRX_QUE_LOCK_WAIT;
|
||||||
|
trx->wait_started = time(NULL);
|
||||||
|
|
||||||
ut_a(que_thr_stop(thr));
|
ut_a(que_thr_stop(thr));
|
||||||
|
|
||||||
|
@ -3432,6 +3513,9 @@ lock_rec_print(
|
||||||
|
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"Suppressing further record lock prints for this page\n");
|
"Suppressing further record lock prints for this page\n");
|
||||||
|
|
||||||
|
mtr_commit(&mtr);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3520,6 +3604,22 @@ lock_print_info(
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"Total number of lock structs in row lock hash table %lu\n",
|
"Total number of lock structs in row lock hash table %lu\n",
|
||||||
lock_get_n_rec_locks());
|
lock_get_n_rec_locks());
|
||||||
|
if (lock_deadlock_found) {
|
||||||
|
|
||||||
|
if ((ulint)(buf_end - buf)
|
||||||
|
< 100 + strlen(lock_latest_err_buf)) {
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf += sprintf(buf, "%s", lock_latest_err_buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (buf_end - buf < 600) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf += sprintf(buf, "LIST OF TRANSACTIONS FOR EACH SESSION:\n");
|
||||||
|
|
||||||
/* First print info on non-active transactions */
|
/* First print info on non-active transactions */
|
||||||
|
|
||||||
|
@ -3588,7 +3688,8 @@ loop:
|
||||||
|
|
||||||
if (trx->que_state == TRX_QUE_LOCK_WAIT) {
|
if (trx->que_state == TRX_QUE_LOCK_WAIT) {
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"------------------TRX IS WAITING FOR THE LOCK:\n");
|
"------- TRX HAS BEEN WAITING %lu SEC FOR THIS LOCK TO BE GRANTED:\n",
|
||||||
|
(ulint)difftime(time(NULL), trx->wait_started));
|
||||||
|
|
||||||
if (lock_get_type(trx->wait_lock) == LOCK_REC) {
|
if (lock_get_type(trx->wait_lock) == LOCK_REC) {
|
||||||
lock_rec_print(buf, trx->wait_lock);
|
lock_rec_print(buf, trx->wait_lock);
|
||||||
|
|
|
@ -270,7 +270,7 @@ part_loop:
|
||||||
|
|
||||||
log->lsn = ut_dulint_add(log->lsn, len);
|
log->lsn = ut_dulint_add(log->lsn, len);
|
||||||
|
|
||||||
/* Initialize the next block header and trailer */
|
/* Initialize the next block header */
|
||||||
log_block_init(log_block + OS_FILE_LOG_BLOCK_SIZE, log->lsn);
|
log_block_init(log_block + OS_FILE_LOG_BLOCK_SIZE, log->lsn);
|
||||||
} else {
|
} else {
|
||||||
log->lsn = ut_dulint_add(log->lsn, len);
|
log->lsn = ut_dulint_add(log->lsn, len);
|
||||||
|
@ -1070,28 +1070,16 @@ log_group_file_header_flush(
|
||||||
}
|
}
|
||||||
|
|
||||||
/**********************************************************
|
/**********************************************************
|
||||||
Stores a 1-byte checksum to the trailer checksum field of a log block
|
Stores a 4-byte checksum to the trailer checksum field of a log block
|
||||||
before writing it to a log file. This checksum is used in recovery to
|
before writing it to a log file. This checksum is used in recovery to
|
||||||
check the consistency of a log block. The checksum is simply the 8 low
|
check the consistency of a log block. */
|
||||||
bits of 1 + the sum of the bytes in the log block except the trailer bytes. */
|
|
||||||
static
|
static
|
||||||
void
|
void
|
||||||
log_block_store_checksum(
|
log_block_store_checksum(
|
||||||
/*=====================*/
|
/*=====================*/
|
||||||
byte* block) /* in/out: pointer to a log block */
|
byte* block) /* in/out: pointer to a log block */
|
||||||
{
|
{
|
||||||
ulint i;
|
log_block_set_checksum(block, log_block_calc_checksum(block));
|
||||||
ulint sum;
|
|
||||||
|
|
||||||
sum = 1;
|
|
||||||
|
|
||||||
for (i = 0; i < OS_FILE_LOG_BLOCK_SIZE - LOG_BLOCK_TRL_SIZE; i++) {
|
|
||||||
sum += (ulint)(*(block + i));
|
|
||||||
}
|
|
||||||
|
|
||||||
mach_write_to_1(block + OS_FILE_LOG_BLOCK_SIZE
|
|
||||||
- LOG_BLOCK_TRL_CHECKSUM,
|
|
||||||
0xFF & sum);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**********************************************************
|
/**********************************************************
|
||||||
|
@ -3113,8 +3101,8 @@ log_print(
|
||||||
|
|
||||||
current_time = time(NULL);
|
current_time = time(NULL);
|
||||||
|
|
||||||
time_elapsed = difftime(current_time, log_sys->last_printout_time);
|
time_elapsed = 0.001 + difftime(current_time,
|
||||||
|
log_sys->last_printout_time);
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"%lu pending log writes, %lu pending chkp writes\n"
|
"%lu pending log writes, %lu pending chkp writes\n"
|
||||||
"%lu log i/o's done, %.2f log i/o's/second\n",
|
"%lu log i/o's done, %.2f log i/o's/second\n",
|
||||||
|
@ -3128,3 +3116,14 @@ log_print(
|
||||||
|
|
||||||
mutex_exit(&(log_sys->mutex));
|
mutex_exit(&(log_sys->mutex));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**************************************************************************
|
||||||
|
Refreshes the statistics used to print per-second averages. */
|
||||||
|
|
||||||
|
void
|
||||||
|
log_refresh_stats(void)
|
||||||
|
/*===================*/
|
||||||
|
{
|
||||||
|
log_sys->n_log_ios_old = log_sys->n_log_ios;
|
||||||
|
log_sys->last_printout_time = time(NULL);
|
||||||
|
}
|
||||||
|
|
|
@ -58,12 +58,16 @@ yet: the variable name is misleading */
|
||||||
|
|
||||||
ibool recv_no_ibuf_operations = FALSE;
|
ibool recv_no_ibuf_operations = FALSE;
|
||||||
|
|
||||||
/* the following counter is used to decide when to print info on
|
/* The following counter is used to decide when to print info on
|
||||||
log scan */
|
log scan */
|
||||||
ulint recv_scan_print_counter = 0;
|
ulint recv_scan_print_counter = 0;
|
||||||
|
|
||||||
ibool recv_is_from_backup = FALSE;
|
ibool recv_is_from_backup = FALSE;
|
||||||
|
ibool recv_is_making_a_backup = FALSE;
|
||||||
|
|
||||||
|
ulint recv_previous_parsed_rec_type = 999999;
|
||||||
|
ulint recv_previous_parsed_rec_offset = 0;
|
||||||
|
ulint recv_previous_parsed_rec_is_multi = 0;
|
||||||
|
|
||||||
/************************************************************
|
/************************************************************
|
||||||
Creates the recovery system. */
|
Creates the recovery system. */
|
||||||
|
@ -124,6 +128,8 @@ recv_sys_init(
|
||||||
|
|
||||||
recv_sys->last_block = ut_align(recv_sys->last_block_buf_start,
|
recv_sys->last_block = ut_align(recv_sys->last_block_buf_start,
|
||||||
OS_FILE_LOG_BLOCK_SIZE);
|
OS_FILE_LOG_BLOCK_SIZE);
|
||||||
|
recv_sys->found_corrupt_log = FALSE;
|
||||||
|
|
||||||
mutex_exit(&(recv_sys->mutex));
|
mutex_exit(&(recv_sys->mutex));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -569,9 +575,9 @@ recv_read_cp_info_for_backup(
|
||||||
}
|
}
|
||||||
|
|
||||||
/**********************************************************
|
/**********************************************************
|
||||||
Checks the 1-byte checksum to the trailer checksum field of a log block.
|
Checks the 4-byte checksum to the trailer checksum field of a log block.
|
||||||
We also accept a log block in the old format where the checksum field
|
We also accept a log block in the old format < InnoDB-3.23.52 where the
|
||||||
contained the highest byte of the log block number. */
|
checksum field contains the log block number. */
|
||||||
static
|
static
|
||||||
ibool
|
ibool
|
||||||
log_block_checksum_is_ok_or_old_format(
|
log_block_checksum_is_ok_or_old_format(
|
||||||
|
@ -580,29 +586,12 @@ log_block_checksum_is_ok_or_old_format(
|
||||||
format of InnoDB version < 3.23.52 */
|
format of InnoDB version < 3.23.52 */
|
||||||
byte* block) /* in: pointer to a log block */
|
byte* block) /* in: pointer to a log block */
|
||||||
{
|
{
|
||||||
ulint i;
|
if (log_block_calc_checksum(block) == log_block_get_checksum(block)) {
|
||||||
ulint sum;
|
|
||||||
|
|
||||||
sum = 1;
|
|
||||||
|
|
||||||
for (i = 0; i < OS_FILE_LOG_BLOCK_SIZE - LOG_BLOCK_TRL_SIZE; i++) {
|
|
||||||
sum += (ulint)(*(block + i));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* printf("Checksum %lu, byte %lu\n", 0xFF & sum,
|
|
||||||
mach_read_from_1(block + OS_FILE_LOG_BLOCK_SIZE
|
|
||||||
- LOG_BLOCK_TRL_CHECKSUM));
|
|
||||||
*/
|
|
||||||
if (mach_read_from_1(block + OS_FILE_LOG_BLOCK_SIZE
|
|
||||||
- LOG_BLOCK_TRL_CHECKSUM)
|
|
||||||
== (0xFF & sum)) {
|
|
||||||
|
|
||||||
return(TRUE);
|
return(TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (((0xFF000000 & log_block_get_hdr_no(block)) >> 24)
|
if (log_block_get_hdr_no(block) == log_block_get_checksum(block)) {
|
||||||
== mach_read_from_1(block + OS_FILE_LOG_BLOCK_SIZE
|
|
||||||
- LOG_BLOCK_TRL_CHECKSUM)) {
|
|
||||||
|
|
||||||
/* We assume the log block is in the format of
|
/* We assume the log block is in the format of
|
||||||
InnoDB version < 3.23.52 and the block is ok */
|
InnoDB version < 3.23.52 and the block is ok */
|
||||||
|
@ -649,23 +638,20 @@ recv_scan_log_seg_for_backup(
|
||||||
|
|
||||||
/* fprintf(stderr, "Log block header no %lu\n", no); */
|
/* fprintf(stderr, "Log block header no %lu\n", no); */
|
||||||
|
|
||||||
if ((no & 0xFFFFFF) != log_block_get_trl_no(log_block)
|
if (no != log_block_convert_lsn_to_no(*scanned_lsn)
|
||||||
|| no != log_block_convert_lsn_to_no(*scanned_lsn)
|
|
||||||
|| !log_block_checksum_is_ok_or_old_format(log_block)) {
|
|| !log_block_checksum_is_ok_or_old_format(log_block)) {
|
||||||
/*
|
/*
|
||||||
printf(
|
printf(
|
||||||
"Log block n:o %lu, trailer n:o %lu, scanned lsn n:o %lu\n",
|
"Log block n:o %lu, scanned lsn n:o %lu\n",
|
||||||
no, log_block_get_trl_no(log_block),
|
no, log_block_convert_lsn_to_no(*scanned_lsn));
|
||||||
log_block_convert_lsn_to_no(*scanned_lsn));
|
|
||||||
*/
|
*/
|
||||||
/* Garbage or an incompletely written log block */
|
/* Garbage or an incompletely written log block */
|
||||||
|
|
||||||
log_block += OS_FILE_LOG_BLOCK_SIZE;
|
log_block += OS_FILE_LOG_BLOCK_SIZE;
|
||||||
/*
|
/*
|
||||||
printf(
|
printf(
|
||||||
"Next log block n:o %lu, trailer n:o %lu\n",
|
"Next log block n:o %lu\n",
|
||||||
log_block_get_hdr_no(log_block),
|
log_block_get_hdr_no(log_block));
|
||||||
log_block_get_trl_no(log_block));
|
|
||||||
*/
|
*/
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -788,14 +774,8 @@ recv_parse_or_apply_log_rec_body(
|
||||||
new_ptr = mlog_parse_string(ptr, end_ptr, page);
|
new_ptr = mlog_parse_string(ptr, end_ptr, page);
|
||||||
} else {
|
} else {
|
||||||
new_ptr = NULL;
|
new_ptr = NULL;
|
||||||
|
|
||||||
fprintf(stderr,
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
"InnoDB: WARNING: the log file may have been corrupt and it\n"
|
|
||||||
"InnoDB: is possible that the log scan did not proceed\n"
|
|
||||||
"InnoDB: far enough in recovery. Please run CHECK TABLE\n"
|
|
||||||
"InnoDB: on your InnoDB tables to check that they are ok!\n"
|
|
||||||
"InnoDB: Corrupt log record type %lu\n",
|
|
||||||
(ulint)type);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ut_ad(!page || new_ptr);
|
ut_ad(!page || new_ptr);
|
||||||
|
@ -1399,18 +1379,30 @@ recv_apply_log_recs_for_backup(
|
||||||
OS_FILE_OPEN,
|
OS_FILE_OPEN,
|
||||||
OS_FILE_READ_WRITE,
|
OS_FILE_READ_WRITE,
|
||||||
&success);
|
&success);
|
||||||
ut_a(success);
|
if (!success) {
|
||||||
|
printf(
|
||||||
|
"InnoDB: Error: cannot open %lu'th data file %s\n", nth_file);
|
||||||
|
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
recv_addr = recv_get_fil_addr_struct(0, i);
|
recv_addr = recv_get_fil_addr_struct(0, i);
|
||||||
|
|
||||||
if (recv_addr != NULL) {
|
if (recv_addr != NULL) {
|
||||||
os_file_read(data_file, page,
|
success = os_file_read(data_file, page,
|
||||||
(nth_page_in_file << UNIV_PAGE_SIZE_SHIFT)
|
(nth_page_in_file << UNIV_PAGE_SIZE_SHIFT)
|
||||||
& 0xFFFFFFFF,
|
& 0xFFFFFFFF,
|
||||||
nth_page_in_file >> (32 - UNIV_PAGE_SIZE_SHIFT),
|
nth_page_in_file >> (32 - UNIV_PAGE_SIZE_SHIFT),
|
||||||
UNIV_PAGE_SIZE);
|
UNIV_PAGE_SIZE);
|
||||||
|
if (!success) {
|
||||||
|
printf(
|
||||||
|
"InnoDB: Error: cannot read page no %lu from %lu'th data file %s\n",
|
||||||
|
nth_page_in_file, nth_file);
|
||||||
|
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
/* We simulate a page read made by the buffer pool,
|
/* We simulate a page read made by the buffer pool,
|
||||||
to make sure recovery works ok. We must init the
|
to make sure recovery works ok. We must init the
|
||||||
block corresponding to buf_pool->frame_zero
|
block corresponding to buf_pool->frame_zero
|
||||||
|
@ -1425,12 +1417,19 @@ recv_apply_log_recs_for_backup(
|
||||||
mach_read_from_8(page + FIL_PAGE_LSN),
|
mach_read_from_8(page + FIL_PAGE_LSN),
|
||||||
0, i);
|
0, i);
|
||||||
|
|
||||||
os_file_write(data_files[nth_file],
|
success = os_file_write(data_files[nth_file],
|
||||||
data_file, page,
|
data_file, page,
|
||||||
(nth_page_in_file << UNIV_PAGE_SIZE_SHIFT)
|
(nth_page_in_file << UNIV_PAGE_SIZE_SHIFT)
|
||||||
& 0xFFFFFFFF,
|
& 0xFFFFFFFF,
|
||||||
nth_page_in_file >> (32 - UNIV_PAGE_SIZE_SHIFT),
|
nth_page_in_file >> (32 - UNIV_PAGE_SIZE_SHIFT),
|
||||||
UNIV_PAGE_SIZE);
|
UNIV_PAGE_SIZE);
|
||||||
|
if (!success) {
|
||||||
|
printf(
|
||||||
|
"InnoDB: Error: cannot write page no %lu to %lu'th data file %s\n",
|
||||||
|
nth_page_in_file, nth_file);
|
||||||
|
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((100 * i) / n_pages_total
|
if ((100 * i) / n_pages_total
|
||||||
|
@ -1679,29 +1678,16 @@ recv_parse_log_rec(
|
||||||
|
|
||||||
new_ptr = mlog_parse_initial_log_record(ptr, end_ptr, type, space,
|
new_ptr = mlog_parse_initial_log_record(ptr, end_ptr, type, space,
|
||||||
page_no);
|
page_no);
|
||||||
|
|
||||||
/* If the operating system writes to the log complete 512-byte
|
|
||||||
blocks, we should not get the warnings below in recovery.
|
|
||||||
A warning means that the header and the trailer appeared ok
|
|
||||||
in a 512-byte block, but in the middle there was something wrong.
|
|
||||||
TODO: (1) add similar warnings in the case there is an incompletely
|
|
||||||
written log record which does not extend to the boundary of a
|
|
||||||
512-byte block. (2) Add a checksum to a log block. */
|
|
||||||
|
|
||||||
if (!new_ptr) {
|
if (!new_ptr) {
|
||||||
|
|
||||||
return(0);
|
return(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check that space id and page_no are sensible */
|
/* Check that space id and page_no are sensible */
|
||||||
|
|
||||||
if (*space != 0 || *page_no > 0x8FFFFFFF) {
|
if (*space != 0 || *page_no > 0x8FFFFFFF) {
|
||||||
fprintf(stderr,
|
|
||||||
"InnoDB: WARNING: the log file may have been corrupt and it\n"
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
"InnoDB: is possible that the log scan did not proceed\n"
|
|
||||||
"InnoDB: far enough in recovery. Please run CHECK TABLE\n"
|
|
||||||
"InnoDB: on your InnoDB tables to check that they are ok!\n"
|
|
||||||
"InnoDB: Corrupt log record type %lu, space id %lu, page no %lu\n",
|
|
||||||
(ulint)(*type), *space, *page_no);
|
|
||||||
|
|
||||||
return(0);
|
return(0);
|
||||||
}
|
}
|
||||||
|
@ -1765,6 +1751,63 @@ recv_check_incomplete_log_recs(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/***********************************************************
|
||||||
|
Prints diagnostic info of corrupt log. */
|
||||||
|
static
|
||||||
|
void
|
||||||
|
recv_report_corrupt_log(
|
||||||
|
/*====================*/
|
||||||
|
byte* ptr, /* in: pointer to corrupt log record */
|
||||||
|
byte type, /* in: type of the record */
|
||||||
|
ulint space, /* in: space id, this may also be garbage */
|
||||||
|
ulint page_no)/* in: page number, this may also be garbage */
|
||||||
|
{
|
||||||
|
char* err_buf;
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: ############### CORRUPT LOG RECORD FOUND\n"
|
||||||
|
"InnoDB: Log record type %lu, space id %lu, page number %lu\n"
|
||||||
|
"InnoDB: Log parsing proceeded successfully up to %lu %lu\n",
|
||||||
|
(ulint)type, space, page_no,
|
||||||
|
ut_dulint_get_high(recv_sys->recovered_lsn),
|
||||||
|
ut_dulint_get_low(recv_sys->recovered_lsn));
|
||||||
|
|
||||||
|
err_buf = ut_malloc(1000000);
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Previous log record type %lu, is multi %lu\n"
|
||||||
|
"InnoDB: Recv offset %lu, prev %lu\n",
|
||||||
|
recv_previous_parsed_rec_type,
|
||||||
|
recv_previous_parsed_rec_is_multi,
|
||||||
|
ptr - recv_sys->buf,
|
||||||
|
recv_previous_parsed_rec_offset);
|
||||||
|
|
||||||
|
if ((ulint)(ptr - recv_sys->buf + 100)
|
||||||
|
> recv_previous_parsed_rec_offset
|
||||||
|
&& (ulint)(ptr - recv_sys->buf + 100
|
||||||
|
- recv_previous_parsed_rec_offset)
|
||||||
|
< 200000) {
|
||||||
|
|
||||||
|
ut_sprintf_buf(err_buf,
|
||||||
|
recv_sys->buf + recv_previous_parsed_rec_offset - 100,
|
||||||
|
ptr - recv_sys->buf + 200 -
|
||||||
|
recv_previous_parsed_rec_offset);
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Hex dump of corrupt log starting 100 bytes before the start\n"
|
||||||
|
"InnoDB: of the previous log rec,\n"
|
||||||
|
"InnoDB: and ending 100 bytes after the start of the corrupt rec:\n%s\n",
|
||||||
|
err_buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
ut_free(err_buf);
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: WARNING: the log file may have been corrupt and it\n"
|
||||||
|
"InnoDB: is possible that the log scan did not proceed\n"
|
||||||
|
"InnoDB: far enough in recovery! Please run CHECK TABLE\n"
|
||||||
|
"InnoDB: on your InnoDB tables to check that they are ok!\n");
|
||||||
|
}
|
||||||
|
|
||||||
/***********************************************************
|
/***********************************************************
|
||||||
Parses log records from a buffer and stores them to a hash table to wait
|
Parses log records from a buffer and stores them to a hash table to wait
|
||||||
merging to file pages. */
|
merging to file pages. */
|
||||||
|
@ -1772,8 +1815,7 @@ static
|
||||||
ibool
|
ibool
|
||||||
recv_parse_log_recs(
|
recv_parse_log_recs(
|
||||||
/*================*/
|
/*================*/
|
||||||
/* out: TRUE if the hash table of parsed log
|
/* out: currently always returns FALSE */
|
||||||
records became full */
|
|
||||||
ibool store_to_hash) /* in: TRUE if the records should be stored
|
ibool store_to_hash) /* in: TRUE if the records should be stored
|
||||||
to the hash table; this is set to FALSE if just
|
to the hash table; this is set to FALSE if just
|
||||||
debug checking is needed */
|
debug checking is needed */
|
||||||
|
@ -1812,8 +1854,13 @@ loop:
|
||||||
|
|
||||||
len = recv_parse_log_rec(ptr, end_ptr, &type, &space,
|
len = recv_parse_log_rec(ptr, end_ptr, &type, &space,
|
||||||
&page_no, &body);
|
&page_no, &body);
|
||||||
if (len == 0) {
|
if (len == 0 || recv_sys->found_corrupt_log) {
|
||||||
|
if (recv_sys->found_corrupt_log) {
|
||||||
|
|
||||||
|
recv_report_corrupt_log(ptr,
|
||||||
|
type, space, page_no);
|
||||||
|
}
|
||||||
|
|
||||||
return(FALSE);
|
return(FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1828,6 +1875,10 @@ loop:
|
||||||
return(FALSE);
|
return(FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
recv_previous_parsed_rec_type = (ulint)type;
|
||||||
|
recv_previous_parsed_rec_offset = recv_sys->recovered_offset;
|
||||||
|
recv_previous_parsed_rec_is_multi = 0;
|
||||||
|
|
||||||
recv_sys->recovered_offset += len;
|
recv_sys->recovered_offset += len;
|
||||||
recv_sys->recovered_lsn = new_recovered_lsn;
|
recv_sys->recovered_lsn = new_recovered_lsn;
|
||||||
|
|
||||||
|
@ -1851,9 +1902,10 @@ loop:
|
||||||
#ifdef UNIV_LOG_DEBUG
|
#ifdef UNIV_LOG_DEBUG
|
||||||
recv_check_incomplete_log_recs(ptr, len);
|
recv_check_incomplete_log_recs(ptr, len);
|
||||||
#endif
|
#endif
|
||||||
recv_update_replicate(type, space, page_no, body,
|
/* recv_update_replicate(type, space, page_no, body,
|
||||||
ptr + len);
|
ptr + len);
|
||||||
recv_compare_replicate(space, page_no);
|
recv_compare_replicate(space, page_no);
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* Check that all the records associated with the single mtr
|
/* Check that all the records associated with the single mtr
|
||||||
|
@ -1865,19 +1917,32 @@ loop:
|
||||||
for (;;) {
|
for (;;) {
|
||||||
len = recv_parse_log_rec(ptr, end_ptr, &type, &space,
|
len = recv_parse_log_rec(ptr, end_ptr, &type, &space,
|
||||||
&page_no, &body);
|
&page_no, &body);
|
||||||
if (len == 0) {
|
if (len == 0 || recv_sys->found_corrupt_log) {
|
||||||
|
|
||||||
return(FALSE);
|
if (recv_sys->found_corrupt_log) {
|
||||||
|
|
||||||
|
recv_report_corrupt_log(ptr,
|
||||||
|
type, space, page_no);
|
||||||
|
}
|
||||||
|
|
||||||
|
return(FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
recv_previous_parsed_rec_type = (ulint)type;
|
||||||
|
recv_previous_parsed_rec_offset
|
||||||
|
= recv_sys->recovered_offset + total_len;
|
||||||
|
recv_previous_parsed_rec_is_multi = 1;
|
||||||
|
|
||||||
if ((!store_to_hash) && (type != MLOG_MULTI_REC_END)) {
|
if ((!store_to_hash) && (type != MLOG_MULTI_REC_END)) {
|
||||||
/* In debug checking, update a replicate page
|
/* In debug checking, update a replicate page
|
||||||
according to the log record */
|
according to the log record */
|
||||||
#ifdef UNIV_LOG_DEBUG
|
#ifdef UNIV_LOG_DEBUG
|
||||||
recv_check_incomplete_log_recs(ptr, len);
|
recv_check_incomplete_log_recs(ptr, len);
|
||||||
#endif
|
#endif
|
||||||
|
/*
|
||||||
recv_update_replicate(type, space, page_no,
|
recv_update_replicate(type, space, page_no,
|
||||||
body, ptr + len);
|
body, ptr + len);
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
if (log_debug_writes) {
|
if (log_debug_writes) {
|
||||||
|
@ -1919,6 +1984,12 @@ loop:
|
||||||
old_lsn = recv_sys->recovered_lsn;
|
old_lsn = recv_sys->recovered_lsn;
|
||||||
len = recv_parse_log_rec(ptr, end_ptr, &type, &space,
|
len = recv_parse_log_rec(ptr, end_ptr, &type, &space,
|
||||||
&page_no, &body);
|
&page_no, &body);
|
||||||
|
if (recv_sys->found_corrupt_log) {
|
||||||
|
|
||||||
|
recv_report_corrupt_log(ptr,
|
||||||
|
type, space, page_no);
|
||||||
|
}
|
||||||
|
|
||||||
ut_a(len != 0);
|
ut_a(len != 0);
|
||||||
ut_a(0 == ((ulint)*ptr & MLOG_SINGLE_REC_FLAG));
|
ut_a(0 == ((ulint)*ptr & MLOG_SINGLE_REC_FLAG));
|
||||||
|
|
||||||
|
@ -1941,7 +2012,7 @@ loop:
|
||||||
page has become identical with the original
|
page has become identical with the original
|
||||||
page */
|
page */
|
||||||
|
|
||||||
recv_compare_replicate(space, page_no);
|
/* recv_compare_replicate(space, page_no); */
|
||||||
}
|
}
|
||||||
|
|
||||||
ptr += len;
|
ptr += len;
|
||||||
|
@ -2095,32 +2166,19 @@ recv_scan_log_recs(
|
||||||
|
|
||||||
/* fprintf(stderr, "Log block header no %lu\n", no); */
|
/* fprintf(stderr, "Log block header no %lu\n", no); */
|
||||||
|
|
||||||
if ((no & 0xFFFFFF) != log_block_get_trl_no(log_block)
|
if (no != log_block_convert_lsn_to_no(scanned_lsn)
|
||||||
|| no != log_block_convert_lsn_to_no(scanned_lsn)
|
|
||||||
|| !log_block_checksum_is_ok_or_old_format(log_block)) {
|
|| !log_block_checksum_is_ok_or_old_format(log_block)) {
|
||||||
|
|
||||||
if ((no & 0xFFFFFF) == log_block_get_trl_no(log_block)
|
if (no == log_block_convert_lsn_to_no(scanned_lsn)
|
||||||
&& no == log_block_convert_lsn_to_no(scanned_lsn)
|
|
||||||
&& !log_block_checksum_is_ok_or_old_format(
|
&& !log_block_checksum_is_ok_or_old_format(
|
||||||
log_block)) {
|
log_block)) {
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
"InnoDB: Log block no %lu at lsn %lu %lu has\n"
|
"InnoDB: Log block no %lu at lsn %lu %lu has\n"
|
||||||
"InnoDB: ok header and trailer, but checksum field contains %lu\n",
|
"InnoDB: ok header, but checksum field contains %lu, should be %lu\n",
|
||||||
no, ut_dulint_get_high(scanned_lsn),
|
no, ut_dulint_get_high(scanned_lsn),
|
||||||
ut_dulint_get_low(scanned_lsn),
|
ut_dulint_get_low(scanned_lsn),
|
||||||
mach_read_from_1(log_block
|
log_block_get_checksum(log_block),
|
||||||
+ OS_FILE_LOG_BLOCK_SIZE
|
log_block_calc_checksum(log_block));
|
||||||
- LOG_BLOCK_TRL_CHECKSUM));
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((no & 0xFFFFFF)
|
|
||||||
!= log_block_get_trl_no(log_block)) {
|
|
||||||
fprintf(stderr,
|
|
||||||
"InnoDB: Log block with header no %lu at lsn %lu %lu has\n"
|
|
||||||
"InnoDB: trailer no %lu\n",
|
|
||||||
no, ut_dulint_get_high(scanned_lsn),
|
|
||||||
ut_dulint_get_low(scanned_lsn),
|
|
||||||
log_block_get_trl_no(log_block));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Garbage or an incompletely written log block */
|
/* Garbage or an incompletely written log block */
|
||||||
|
@ -2192,11 +2250,14 @@ recv_scan_log_recs(
|
||||||
>= RECV_PARSING_BUF_SIZE) {
|
>= RECV_PARSING_BUF_SIZE) {
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
"InnoDB: Error: log parsing buffer overflow. Recovery may have failed!\n");
|
"InnoDB: Error: log parsing buffer overflow. Recovery may have failed!\n");
|
||||||
finished = TRUE;
|
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
} else if (!recv_sys->found_corrupt_log) {
|
||||||
|
more_data = recv_sys_add_to_parsing_buf(
|
||||||
|
log_block, scanned_lsn);
|
||||||
}
|
}
|
||||||
|
|
||||||
more_data = recv_sys_add_to_parsing_buf(log_block,
|
|
||||||
scanned_lsn);
|
|
||||||
recv_sys->scanned_lsn = scanned_lsn;
|
recv_sys->scanned_lsn = scanned_lsn;
|
||||||
recv_sys->scanned_checkpoint_no =
|
recv_sys->scanned_checkpoint_no =
|
||||||
log_block_get_checkpoint_no(log_block);
|
log_block_get_checkpoint_no(log_block);
|
||||||
|
@ -2213,7 +2274,8 @@ recv_scan_log_recs(
|
||||||
|
|
||||||
*group_scanned_lsn = scanned_lsn;
|
*group_scanned_lsn = scanned_lsn;
|
||||||
|
|
||||||
if (recv_needed_recovery || recv_is_from_backup) {
|
if (recv_needed_recovery
|
||||||
|
|| (recv_is_from_backup && !recv_is_making_a_backup)) {
|
||||||
recv_scan_print_counter++;
|
recv_scan_print_counter++;
|
||||||
|
|
||||||
if (finished || (recv_scan_print_counter % 80 == 0)) {
|
if (finished || (recv_scan_print_counter % 80 == 0)) {
|
||||||
|
@ -2225,7 +2287,7 @@ recv_scan_log_recs(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (more_data) {
|
if (more_data && !recv_sys->found_corrupt_log) {
|
||||||
/* Try to parse more log records */
|
/* Try to parse more log records */
|
||||||
|
|
||||||
recv_parse_log_recs(store_to_hash);
|
recv_parse_log_recs(store_to_hash);
|
||||||
|
@ -2602,6 +2664,17 @@ recv_recovery_from_checkpoint_finish(void)
|
||||||
trx_sys_print_mysql_binlog_offset();
|
trx_sys_print_mysql_binlog_offset();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (recv_sys->found_corrupt_log) {
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: WARNING: the log file may have been corrupt and it\n"
|
||||||
|
"InnoDB: is possible that the log scan or parsing did not proceed\n"
|
||||||
|
"InnoDB: far enough in recovery. Please run CHECK TABLE\n"
|
||||||
|
"InnoDB: on your InnoDB tables to check that they are ok!\n"
|
||||||
|
"InnoDB: It may be safest to recover your InnoDB database from\n"
|
||||||
|
"InnoDB: a backup!\n");
|
||||||
|
}
|
||||||
|
|
||||||
/* Free the resources of the recovery system */
|
/* Free the resources of the recovery system */
|
||||||
|
|
||||||
recv_recovery_on = FALSE;
|
recv_recovery_on = FALSE;
|
||||||
|
|
|
@ -668,7 +668,7 @@ mem_print_info_low(
|
||||||
|
|
||||||
mem_pool_print_info(outfile, mem_comm_pool);
|
mem_pool_print_info(outfile, mem_comm_pool);
|
||||||
|
|
||||||
mem_validate();
|
/* mem_validate(); */
|
||||||
|
|
||||||
/* fclose(outfile); */
|
/* fclose(outfile); */
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -251,6 +251,7 @@ mem_pool_fill_free_list(
|
||||||
mem_area_t* area;
|
mem_area_t* area;
|
||||||
mem_area_t* area2;
|
mem_area_t* area2;
|
||||||
ibool ret;
|
ibool ret;
|
||||||
|
char err_buf[500];
|
||||||
|
|
||||||
ut_ad(mutex_own(&(pool->mutex)));
|
ut_ad(mutex_own(&(pool->mutex)));
|
||||||
|
|
||||||
|
@ -279,15 +280,34 @@ mem_pool_fill_free_list(
|
||||||
area = UT_LIST_GET_FIRST(pool->free_list[i + 1]);
|
area = UT_LIST_GET_FIRST(pool->free_list[i + 1]);
|
||||||
|
|
||||||
if (area == NULL) {
|
if (area == NULL) {
|
||||||
|
if (UT_LIST_GET_LEN(pool->free_list[i + 1]) > 0) {
|
||||||
|
ut_print_timestamp(stderr);
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
" InnoDB: Error: mem pool free list %lu length is %lu\n"
|
||||||
|
"InnoDB: though the list is empty!\n",
|
||||||
|
i + 1, UT_LIST_GET_LEN(pool->free_list[i + 1]));
|
||||||
|
}
|
||||||
|
|
||||||
ret = mem_pool_fill_free_list(i + 1, pool);
|
ret = mem_pool_fill_free_list(i + 1, pool);
|
||||||
|
|
||||||
if (ret == FALSE) {
|
if (ret == FALSE) {
|
||||||
|
|
||||||
return(FALSE);
|
return(FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
area = UT_LIST_GET_FIRST(pool->free_list[i + 1]);
|
area = UT_LIST_GET_FIRST(pool->free_list[i + 1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (UT_LIST_GET_LEN(pool->free_list[i + 1]) == 0) {
|
||||||
|
ut_sprintf_buf(err_buf, ((byte*)area) - 50, 100);
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: Removing element from mem pool free list %lu\n"
|
||||||
|
"InnoDB: though the list length is 0! Dump of 100 bytes around element:\n%s\n",
|
||||||
|
i + 1, err_buf);
|
||||||
|
ut_a(0);
|
||||||
|
}
|
||||||
|
|
||||||
UT_LIST_REMOVE(free_list, pool->free_list[i + 1], area);
|
UT_LIST_REMOVE(free_list, pool->free_list[i + 1], area);
|
||||||
|
|
||||||
area2 = (mem_area_t*)(((byte*)area) + ut_2_exp(i));
|
area2 = (mem_area_t*)(((byte*)area) + ut_2_exp(i));
|
||||||
|
@ -320,6 +340,7 @@ mem_area_alloc(
|
||||||
mem_area_t* area;
|
mem_area_t* area;
|
||||||
ulint n;
|
ulint n;
|
||||||
ibool ret;
|
ibool ret;
|
||||||
|
char err_buf[500];
|
||||||
|
|
||||||
n = ut_2_log(ut_max(size + MEM_AREA_EXTRA_SIZE, MEM_AREA_MIN_SIZE));
|
n = ut_2_log(ut_max(size + MEM_AREA_EXTRA_SIZE, MEM_AREA_MIN_SIZE));
|
||||||
|
|
||||||
|
@ -342,7 +363,24 @@ mem_area_alloc(
|
||||||
area = UT_LIST_GET_FIRST(pool->free_list[n]);
|
area = UT_LIST_GET_FIRST(pool->free_list[n]);
|
||||||
}
|
}
|
||||||
|
|
||||||
ut_a(mem_area_get_free(area));
|
if (!mem_area_get_free(area)) {
|
||||||
|
ut_sprintf_buf(err_buf, ((byte*)area) - 50, 100);
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: Removing element from mem pool free list %lu though the\n"
|
||||||
|
"InnoDB: element is not marked free! Dump of 100 bytes around element:\n%s\n",
|
||||||
|
n, err_buf);
|
||||||
|
ut_a(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (UT_LIST_GET_LEN(pool->free_list[n]) == 0) {
|
||||||
|
ut_sprintf_buf(err_buf, ((byte*)area) - 50, 100);
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: Removing element from mem pool free list %lu\n"
|
||||||
|
"InnoDB: though the list length is 0! Dump of 100 bytes around element:\n%s\n",
|
||||||
|
n, err_buf);
|
||||||
|
ut_a(0);
|
||||||
|
}
|
||||||
|
|
||||||
ut_ad(mem_area_get_size(area) == ut_2_exp(n));
|
ut_ad(mem_area_get_size(area) == ut_2_exp(n));
|
||||||
|
|
||||||
mem_area_set_free(area, FALSE);
|
mem_area_set_free(area, FALSE);
|
||||||
|
@ -413,6 +451,7 @@ mem_area_free(
|
||||||
void* new_ptr;
|
void* new_ptr;
|
||||||
ulint size;
|
ulint size;
|
||||||
ulint n;
|
ulint n;
|
||||||
|
char err_buf[500];
|
||||||
|
|
||||||
if (mem_out_of_mem_err_msg_count > 0) {
|
if (mem_out_of_mem_err_msg_count > 0) {
|
||||||
/* It may be that the area was really allocated from the
|
/* It may be that the area was really allocated from the
|
||||||
|
@ -429,10 +468,18 @@ mem_area_free(
|
||||||
|
|
||||||
area = (mem_area_t*) (((byte*)ptr) - MEM_AREA_EXTRA_SIZE);
|
area = (mem_area_t*) (((byte*)ptr) - MEM_AREA_EXTRA_SIZE);
|
||||||
|
|
||||||
|
if (mem_area_get_free(area)) {
|
||||||
|
ut_sprintf_buf(err_buf, ((byte*)area) - 50, 100);
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: Freeing element to mem pool free list though the\n"
|
||||||
|
"InnoDB: element is marked free! Dump of 100 bytes around element:\n%s\n",
|
||||||
|
err_buf);
|
||||||
|
ut_a(0);
|
||||||
|
}
|
||||||
|
|
||||||
size = mem_area_get_size(area);
|
size = mem_area_get_size(area);
|
||||||
|
|
||||||
ut_ad(size != 0);
|
ut_ad(size != 0);
|
||||||
ut_a(!mem_area_get_free(area));
|
|
||||||
|
|
||||||
#ifdef UNIV_LIGHT_MEM_DEBUG
|
#ifdef UNIV_LIGHT_MEM_DEBUG
|
||||||
if (((byte*)area) + size < pool->buf + pool->size) {
|
if (((byte*)area) + size < pool->buf + pool->size) {
|
||||||
|
|
|
@ -14,6 +14,7 @@ Created 12/7/1995 Heikki Tuuri
|
||||||
|
|
||||||
#include "buf0buf.h"
|
#include "buf0buf.h"
|
||||||
#include "dict0boot.h"
|
#include "dict0boot.h"
|
||||||
|
#include "log0recv.h"
|
||||||
|
|
||||||
/************************************************************
|
/************************************************************
|
||||||
Catenates n bytes to the mtr log. */
|
Catenates n bytes to the mtr log. */
|
||||||
|
@ -121,7 +122,7 @@ byte*
|
||||||
mlog_parse_nbytes(
|
mlog_parse_nbytes(
|
||||||
/*==============*/
|
/*==============*/
|
||||||
/* out: parsed record end, NULL if not a complete
|
/* out: parsed record end, NULL if not a complete
|
||||||
record */
|
record or a corrupt record */
|
||||||
ulint type, /* in: log record type: MLOG_1BYTE, ... */
|
ulint type, /* in: log record type: MLOG_1BYTE, ... */
|
||||||
byte* ptr, /* in: buffer */
|
byte* ptr, /* in: buffer */
|
||||||
byte* end_ptr,/* in: buffer end */
|
byte* end_ptr,/* in: buffer end */
|
||||||
|
@ -141,6 +142,12 @@ mlog_parse_nbytes(
|
||||||
offset = mach_read_from_2(ptr);
|
offset = mach_read_from_2(ptr);
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
|
|
||||||
|
if (offset >= UNIV_PAGE_SIZE) {
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
if (type == MLOG_8BYTES) {
|
if (type == MLOG_8BYTES) {
|
||||||
ptr = mach_dulint_parse_compressed(ptr, end_ptr, &dval);
|
ptr = mach_dulint_parse_compressed(ptr, end_ptr, &dval);
|
||||||
|
|
||||||
|
@ -163,13 +170,33 @@ mlog_parse_nbytes(
|
||||||
return(NULL);
|
return(NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (type == MLOG_1BYTE) {
|
||||||
|
if (val > 0xFF) {
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
} else if (type == MLOG_2BYTES) {
|
||||||
|
if (val > 0xFFFF) {
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (type != MLOG_4BYTES) {
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (page) {
|
if (page) {
|
||||||
if (type == MLOG_1BYTE) {
|
if (type == MLOG_1BYTE) {
|
||||||
mach_write_to_1(page + offset, val);
|
mach_write_to_1(page + offset, val);
|
||||||
} else if (type == MLOG_2BYTES) {
|
} else if (type == MLOG_2BYTES) {
|
||||||
mach_write_to_2(page + offset, val);
|
mach_write_to_2(page + offset, val);
|
||||||
} else {
|
} else {
|
||||||
ut_ad(type == MLOG_4BYTES);
|
ut_a(type == MLOG_4BYTES);
|
||||||
mach_write_to_4(page + offset, val);
|
mach_write_to_4(page + offset, val);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -338,7 +365,11 @@ mlog_parse_string(
|
||||||
offset = mach_read_from_2(ptr);
|
offset = mach_read_from_2(ptr);
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
|
|
||||||
ut_a(offset < UNIV_PAGE_SIZE);
|
if (offset >= UNIV_PAGE_SIZE) {
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
len = mach_read_from_2(ptr);
|
len = mach_read_from_2(ptr);
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
|
|
|
@ -22,7 +22,7 @@ Created 10/21/1995 Heikki Tuuri
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* This specifies the file permissions InnoDB uses when it craetes files in
|
/* This specifies the file permissions InnoDB uses when it creates files in
|
||||||
Unix; the value of os_innodb_umask is initialized in ha_innodb.cc to
|
Unix; the value of os_innodb_umask is initialized in ha_innodb.cc to
|
||||||
my_umask */
|
my_umask */
|
||||||
|
|
||||||
|
@ -2483,7 +2483,7 @@ loop:
|
||||||
buf += sprintf(buf, "\n");
|
buf += sprintf(buf, "\n");
|
||||||
|
|
||||||
current_time = time(NULL);
|
current_time = time(NULL);
|
||||||
time_elapsed = difftime(current_time, os_last_printout);
|
time_elapsed = 0.001 + difftime(current_time, os_last_printout);
|
||||||
|
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"Pending flushes (fsync) log: %lu; buffer pool: %lu\n",
|
"Pending flushes (fsync) log: %lu; buffer pool: %lu\n",
|
||||||
|
@ -2517,6 +2517,21 @@ loop:
|
||||||
os_last_printout = current_time;
|
os_last_printout = current_time;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**************************************************************************
|
||||||
|
Refreshes the statistics used to print per-second averages. */
|
||||||
|
|
||||||
|
void
|
||||||
|
os_aio_refresh_stats(void)
|
||||||
|
/*======================*/
|
||||||
|
{
|
||||||
|
os_n_file_reads_old = os_n_file_reads;
|
||||||
|
os_n_file_writes_old = os_n_file_writes;
|
||||||
|
os_n_fsyncs_old = os_n_fsyncs;
|
||||||
|
os_bytes_read_since_printout = 0;
|
||||||
|
|
||||||
|
os_last_printout = time(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
Checks that all slots in the system have been freed, that is, there are
|
Checks that all slots in the system have been freed, that is, there are
|
||||||
no pending io operations. */
|
no pending io operations. */
|
||||||
|
|
|
@ -18,26 +18,63 @@ Created 9/8/1995 Heikki Tuuri
|
||||||
|
|
||||||
#include "srv0srv.h"
|
#include "srv0srv.h"
|
||||||
|
|
||||||
|
/*******************************************************************
|
||||||
|
Compares two threads or thread ids for equality */
|
||||||
|
|
||||||
|
ibool
|
||||||
|
os_thread_eq(
|
||||||
|
/*=========*/
|
||||||
|
/* out: TRUE if equal */
|
||||||
|
os_thread_t a, /* in: OS thread or thread id */
|
||||||
|
os_thread_t b) /* in: OS thread or thread id */
|
||||||
|
{
|
||||||
|
#ifdef __WIN__
|
||||||
|
if (a == b) {
|
||||||
|
return(TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
return(FALSE);
|
||||||
|
#else
|
||||||
|
if (pthread_equal(a, b)) {
|
||||||
|
return(TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
return(FALSE);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/********************************************************************
|
||||||
|
Converts an OS thread or thread id to a ulint. It is NOT guaranteed that
|
||||||
|
the ulint is unique for the thread though! */
|
||||||
|
|
||||||
|
ulint
|
||||||
|
os_thread_pf(
|
||||||
|
/*=========*/
|
||||||
|
os_thread_t a)
|
||||||
|
{
|
||||||
|
#ifdef UNIV_HPUX
|
||||||
|
/* In HP-UX a pthread_t is a struct of 3 fields: field1, field2,
|
||||||
|
field3. We do not know if field1 determines the thread uniquely. */
|
||||||
|
|
||||||
|
return((ulint)(a.field1));
|
||||||
|
#else
|
||||||
|
return((ulint)a);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
Returns the thread identifier of current thread. */
|
Returns the thread identifier of current thread. Currently the thread
|
||||||
|
identifier is the thread handle itself. Note that in HP-UX pthread_t is
|
||||||
|
a struct of 3 fields. */
|
||||||
|
|
||||||
os_thread_id_t
|
os_thread_id_t
|
||||||
os_thread_get_curr_id(void)
|
os_thread_get_curr_id(void)
|
||||||
/*=======================*/
|
/*=======================*/
|
||||||
{
|
{
|
||||||
#ifdef __WIN__
|
#ifdef __WIN__
|
||||||
return(GetCurrentThreadId());
|
return(GetCurrentThread());
|
||||||
#else
|
#else
|
||||||
pthread_t pthr;
|
return(pthread_self());
|
||||||
|
|
||||||
pthr = pthread_self();
|
|
||||||
|
|
||||||
/* TODO: in the future we have to change os_thread_id
|
|
||||||
to pthread_t; the following cast may work in a wrong way on some
|
|
||||||
systems if pthread_t is a struct; this is just a quick fix
|
|
||||||
for HP-UX to eliminate a compiler warning */
|
|
||||||
|
|
||||||
return(*(os_thread_id_t*)((void*) (&pthr)));
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,7 +108,6 @@ os_thread_create(
|
||||||
arg,
|
arg,
|
||||||
0, /* thread runs immediately */
|
0, /* thread runs immediately */
|
||||||
thread_id);
|
thread_id);
|
||||||
ut_a(thread);
|
|
||||||
|
|
||||||
if (srv_set_thread_priorities) {
|
if (srv_set_thread_priorities) {
|
||||||
|
|
||||||
|
@ -108,7 +144,7 @@ Returns handle to the current thread. */
|
||||||
|
|
||||||
os_thread_t
|
os_thread_t
|
||||||
os_thread_get_curr(void)
|
os_thread_get_curr(void)
|
||||||
/*=======================*/
|
/*====================*/
|
||||||
{
|
{
|
||||||
#ifdef __WIN__
|
#ifdef __WIN__
|
||||||
return(GetCurrentThread());
|
return(GetCurrentThread());
|
||||||
|
@ -116,18 +152,6 @@ os_thread_get_curr(void)
|
||||||
return(pthread_self());
|
return(pthread_self());
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
Converts a thread id to a ulint. */
|
|
||||||
|
|
||||||
ulint
|
|
||||||
os_thread_conv_id_to_ulint(
|
|
||||||
/*=======================*/
|
|
||||||
/* out: converted to ulint */
|
|
||||||
os_thread_id_t id) /* in: thread id */
|
|
||||||
{
|
|
||||||
return((ulint)id);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
Advises the os to give up remainder of the thread's time slice. */
|
Advises the os to give up remainder of the thread's time slice. */
|
||||||
|
|
|
@ -13,6 +13,7 @@ Created 10/4/1994 Heikki Tuuri
|
||||||
|
|
||||||
#include "rem0cmp.h"
|
#include "rem0cmp.h"
|
||||||
#include "mtr0log.h"
|
#include "mtr0log.h"
|
||||||
|
#include "log0recv.h"
|
||||||
|
|
||||||
ulint page_cur_short_succ = 0;
|
ulint page_cur_short_succ = 0;
|
||||||
|
|
||||||
|
@ -481,6 +482,9 @@ page_cur_insert_rec_write_log(
|
||||||
|
|
||||||
/* Write the mismatch index */
|
/* Write the mismatch index */
|
||||||
log_ptr += mach_write_compressed(log_ptr, i);
|
log_ptr += mach_write_compressed(log_ptr, i);
|
||||||
|
|
||||||
|
ut_a(i < UNIV_PAGE_SIZE);
|
||||||
|
ut_a(extra_size < UNIV_PAGE_SIZE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Write to the log the inserted index record end segment which
|
/* Write to the log the inserted index record end segment which
|
||||||
|
@ -533,6 +537,13 @@ page_cur_parse_insert_rec(
|
||||||
}
|
}
|
||||||
|
|
||||||
offset = mach_read_from_2(ptr);
|
offset = mach_read_from_2(ptr);
|
||||||
|
|
||||||
|
if (offset >= UNIV_PAGE_SIZE) {
|
||||||
|
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
}
|
}
|
||||||
|
@ -546,6 +557,12 @@ page_cur_parse_insert_rec(
|
||||||
|
|
||||||
extra_info_yes = end_seg_len & 0x1;
|
extra_info_yes = end_seg_len & 0x1;
|
||||||
end_seg_len = end_seg_len / 2;
|
end_seg_len = end_seg_len / 2;
|
||||||
|
|
||||||
|
if (end_seg_len >= UNIV_PAGE_SIZE) {
|
||||||
|
recv_sys->found_corrupt_log = TRUE;
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
if (extra_info_yes) {
|
if (extra_info_yes) {
|
||||||
/* Read the info bits */
|
/* Read the info bits */
|
||||||
|
@ -565,12 +582,16 @@ page_cur_parse_insert_rec(
|
||||||
return(NULL);
|
return(NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ut_a(origin_offset < UNIV_PAGE_SIZE);
|
||||||
|
|
||||||
ptr = mach_parse_compressed(ptr, end_ptr, &mismatch_index);
|
ptr = mach_parse_compressed(ptr, end_ptr, &mismatch_index);
|
||||||
|
|
||||||
if (ptr == NULL) {
|
if (ptr == NULL) {
|
||||||
|
|
||||||
return(NULL);
|
return(NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ut_a(mismatch_index < UNIV_PAGE_SIZE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (end_ptr < ptr + end_seg_len) {
|
if (end_ptr < ptr + end_seg_len) {
|
||||||
|
@ -607,7 +628,6 @@ page_cur_parse_insert_rec(
|
||||||
/* Build the inserted record to buf */
|
/* Build the inserted record to buf */
|
||||||
|
|
||||||
ut_a(mismatch_index < UNIV_PAGE_SIZE);
|
ut_a(mismatch_index < UNIV_PAGE_SIZE);
|
||||||
ut_a(end_seg_len < UNIV_PAGE_SIZE);
|
|
||||||
|
|
||||||
ut_memcpy(buf, rec_get_start(cursor_rec), mismatch_index);
|
ut_memcpy(buf, rec_get_start(cursor_rec), mismatch_index);
|
||||||
ut_memcpy(buf + mismatch_index, ptr, end_seg_len);
|
ut_memcpy(buf + mismatch_index, ptr, end_seg_len);
|
||||||
|
@ -1010,6 +1030,8 @@ page_cur_parse_delete_rec(
|
||||||
offset = mach_read_from_2(ptr);
|
offset = mach_read_from_2(ptr);
|
||||||
ptr += 2;
|
ptr += 2;
|
||||||
|
|
||||||
|
ut_a(offset <= UNIV_PAGE_SIZE);
|
||||||
|
|
||||||
if (page) {
|
if (page) {
|
||||||
page_cur_position(page + offset, &cursor);
|
page_cur_position(page + offset, &cursor);
|
||||||
|
|
||||||
|
|
|
@ -595,6 +595,11 @@ row_lock_table_autoinc_for_mysql(
|
||||||
ut_ad(trx);
|
ut_ad(trx);
|
||||||
ut_ad(trx->mysql_thread_id == os_thread_get_curr_id());
|
ut_ad(trx->mysql_thread_id == os_thread_get_curr_id());
|
||||||
|
|
||||||
|
if (trx->auto_inc_lock) {
|
||||||
|
|
||||||
|
return(DB_SUCCESS);
|
||||||
|
}
|
||||||
|
|
||||||
trx->op_info = (char *) "setting auto-inc lock";
|
trx->op_info = (char *) "setting auto-inc lock";
|
||||||
|
|
||||||
if (node == NULL) {
|
if (node == NULL) {
|
||||||
|
|
|
@ -58,6 +58,7 @@ row_vers_impl_x_locked_off_kernel(
|
||||||
ibool rec_del;
|
ibool rec_del;
|
||||||
ulint err;
|
ulint err;
|
||||||
mtr_t mtr;
|
mtr_t mtr;
|
||||||
|
char err_buf[1000];
|
||||||
|
|
||||||
ut_ad(mutex_own(&kernel_mutex));
|
ut_ad(mutex_own(&kernel_mutex));
|
||||||
ut_ad(!rw_lock_own(&(purge_sys->latch), RW_LOCK_SHARED));
|
ut_ad(!rw_lock_own(&(purge_sys->latch), RW_LOCK_SHARED));
|
||||||
|
@ -74,7 +75,26 @@ row_vers_impl_x_locked_off_kernel(
|
||||||
|
|
||||||
clust_rec = row_get_clust_rec(BTR_SEARCH_LEAF, rec, index,
|
clust_rec = row_get_clust_rec(BTR_SEARCH_LEAF, rec, index,
|
||||||
&clust_index, &mtr);
|
&clust_index, &mtr);
|
||||||
ut_a(clust_rec);
|
if (!clust_rec) {
|
||||||
|
rec_sprintf(err_buf, 900, rec);
|
||||||
|
|
||||||
|
ut_print_timestamp(stderr);
|
||||||
|
fprintf(stderr,
|
||||||
|
" InnoDB: Error: cannot find the clustered index record\n"
|
||||||
|
"InnoDB: for a secondary index record in table %s index %s.\n"
|
||||||
|
"InnoDB: Secondary index record %s.\n"
|
||||||
|
"InnoDB: The table is probably corrupt. Please run CHECK TABLE on it.\n"
|
||||||
|
"InnoDB: You can try to repair the table by dump + drop + reimport.\n"
|
||||||
|
"InnoDB: Send a detailed bug report to mysql@lists.mysql.com.\n",
|
||||||
|
index->table_name, index->name, err_buf);
|
||||||
|
mutex_enter(&kernel_mutex);
|
||||||
|
mtr_commit(&mtr);
|
||||||
|
|
||||||
|
/* We assume there is no lock on the record, though this
|
||||||
|
is not certain because the table is apparently corrupt */
|
||||||
|
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
trx_id = row_get_rec_trx_id(clust_rec, clust_index);
|
trx_id = row_get_rec_trx_id(clust_rec, clust_index);
|
||||||
|
|
||||||
|
|
|
@ -2024,7 +2024,7 @@ srv_table_reserve_slot_for_mysql(void)
|
||||||
|
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
"Slot %lu: thread id %lu, type %lu, in use %lu, susp %lu, time %lu\n",
|
"Slot %lu: thread id %lu, type %lu, in use %lu, susp %lu, time %lu\n",
|
||||||
i, (ulint)(slot->id),
|
i, os_thread_pf(slot->id),
|
||||||
slot->type, slot->in_use,
|
slot->type, slot->in_use,
|
||||||
slot->suspended,
|
slot->suspended,
|
||||||
(ulint)difftime(ut_time(), slot->suspend_time));
|
(ulint)difftime(ut_time(), slot->suspend_time));
|
||||||
|
@ -2168,6 +2168,34 @@ srv_release_mysql_thread_if_suspended(
|
||||||
/* not found */
|
/* not found */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**********************************************************************
|
||||||
|
Refreshes the values used to calculate per-second averages. */
|
||||||
|
static
|
||||||
|
void
|
||||||
|
srv_refresh_innodb_monitor_stats(void)
|
||||||
|
/*==================================*/
|
||||||
|
{
|
||||||
|
mutex_enter(&srv_innodb_monitor_mutex);
|
||||||
|
|
||||||
|
srv_last_monitor_time = time(NULL);
|
||||||
|
|
||||||
|
os_aio_refresh_stats();
|
||||||
|
|
||||||
|
btr_cur_n_sea_old = btr_cur_n_sea;
|
||||||
|
btr_cur_n_non_sea_old = btr_cur_n_non_sea;
|
||||||
|
|
||||||
|
log_refresh_stats();
|
||||||
|
|
||||||
|
buf_refresh_io_stats();
|
||||||
|
|
||||||
|
srv_n_rows_inserted_old = srv_n_rows_inserted;
|
||||||
|
srv_n_rows_updated_old = srv_n_rows_updated;
|
||||||
|
srv_n_rows_deleted_old = srv_n_rows_deleted;
|
||||||
|
srv_n_rows_read_old = srv_n_rows_read;
|
||||||
|
|
||||||
|
mutex_exit(&srv_innodb_monitor_mutex);
|
||||||
|
}
|
||||||
|
|
||||||
/**********************************************************************
|
/**********************************************************************
|
||||||
Sprintfs to a buffer the output of the InnoDB Monitor. */
|
Sprintfs to a buffer the output of the InnoDB Monitor. */
|
||||||
|
|
||||||
|
@ -2205,7 +2233,7 @@ srv_sprintf_innodb_monitor(
|
||||||
"=====================================\n");
|
"=====================================\n");
|
||||||
|
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"Per second values calculated from the last %lu seconds\n",
|
"Per second averages calculated from the last %lu seconds\n",
|
||||||
(ulint)time_elapsed);
|
(ulint)time_elapsed);
|
||||||
|
|
||||||
buf += sprintf(buf, "----------\n"
|
buf += sprintf(buf, "----------\n"
|
||||||
|
@ -2242,8 +2270,8 @@ srv_sprintf_innodb_monitor(
|
||||||
/ time_elapsed,
|
/ time_elapsed,
|
||||||
(btr_cur_n_non_sea - btr_cur_n_non_sea_old)
|
(btr_cur_n_non_sea - btr_cur_n_non_sea_old)
|
||||||
/ time_elapsed);
|
/ time_elapsed);
|
||||||
btr_cur_n_sea_old = btr_cur_n_sea;
|
btr_cur_n_sea_old = btr_cur_n_sea;
|
||||||
btr_cur_n_non_sea_old = btr_cur_n_non_sea;
|
btr_cur_n_non_sea_old = btr_cur_n_non_sea;
|
||||||
|
|
||||||
buf += sprintf(buf,"---\n"
|
buf += sprintf(buf,"---\n"
|
||||||
"LOG\n"
|
"LOG\n"
|
||||||
|
@ -2285,10 +2313,10 @@ srv_sprintf_innodb_monitor(
|
||||||
(srv_n_rows_read - srv_n_rows_read_old)
|
(srv_n_rows_read - srv_n_rows_read_old)
|
||||||
/ time_elapsed);
|
/ time_elapsed);
|
||||||
|
|
||||||
srv_n_rows_inserted_old = srv_n_rows_inserted;
|
srv_n_rows_inserted_old = srv_n_rows_inserted;
|
||||||
srv_n_rows_updated_old = srv_n_rows_updated;
|
srv_n_rows_updated_old = srv_n_rows_updated;
|
||||||
srv_n_rows_deleted_old = srv_n_rows_deleted;
|
srv_n_rows_deleted_old = srv_n_rows_deleted;
|
||||||
srv_n_rows_read_old = srv_n_rows_read;
|
srv_n_rows_read_old = srv_n_rows_read;
|
||||||
|
|
||||||
buf += sprintf(buf, "----------------------------\n"
|
buf += sprintf(buf, "----------------------------\n"
|
||||||
"END OF INNODB MONITOR OUTPUT\n"
|
"END OF INNODB MONITOR OUTPUT\n"
|
||||||
|
@ -2331,7 +2359,7 @@ loop:
|
||||||
/* When someone is waiting for a lock, we wake up every second
|
/* When someone is waiting for a lock, we wake up every second
|
||||||
and check if a timeout has passed for a lock wait */
|
and check if a timeout has passed for a lock wait */
|
||||||
|
|
||||||
os_thread_sleep(1000000);
|
os_thread_sleep(1000000);
|
||||||
|
|
||||||
/* In case mutex_exit is not a memory barrier, it is
|
/* In case mutex_exit is not a memory barrier, it is
|
||||||
theoretically possible some threads are left waiting though
|
theoretically possible some threads are left waiting though
|
||||||
|
@ -2348,9 +2376,9 @@ loop:
|
||||||
|
|
||||||
if (srv_print_innodb_monitor) {
|
if (srv_print_innodb_monitor) {
|
||||||
|
|
||||||
buf = mem_alloc(100000);
|
buf = mem_alloc(100000);
|
||||||
|
|
||||||
srv_sprintf_innodb_monitor(buf, 100000);
|
srv_sprintf_innodb_monitor(buf, 100000);
|
||||||
|
|
||||||
printf("%s", buf);
|
printf("%s", buf);
|
||||||
|
|
||||||
|
@ -2481,12 +2509,30 @@ srv_error_monitor_thread(
|
||||||
void* arg) /* in: a dummy parameter required by
|
void* arg) /* in: a dummy parameter required by
|
||||||
os_thread_create */
|
os_thread_create */
|
||||||
{
|
{
|
||||||
|
ulint cnt = 0;
|
||||||
|
|
||||||
UT_NOT_USED(arg);
|
UT_NOT_USED(arg);
|
||||||
loop:
|
loop:
|
||||||
srv_error_monitor_active = TRUE;
|
srv_error_monitor_active = TRUE;
|
||||||
|
|
||||||
os_thread_sleep(10000000);
|
cnt++;
|
||||||
|
|
||||||
|
os_thread_sleep(2000000);
|
||||||
|
|
||||||
|
if (difftime(time(NULL), srv_last_monitor_time) > 60) {
|
||||||
|
/* We referesh InnoDB Monitor values so that averages are
|
||||||
|
printed from at most 60 last seconds */
|
||||||
|
|
||||||
|
srv_refresh_innodb_monitor_stats();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* mem_print_new_info();
|
||||||
|
|
||||||
|
if (cnt % 10 == 0) {
|
||||||
|
|
||||||
|
mem_print_info();
|
||||||
|
}
|
||||||
|
*/
|
||||||
sync_array_print_long_waits();
|
sync_array_print_long_waits();
|
||||||
|
|
||||||
/* Flush stdout and stderr so that a database user gets their output
|
/* Flush stdout and stderr so that a database user gets their output
|
||||||
|
|
|
@ -74,6 +74,12 @@ ulint ios;
|
||||||
ulint n[SRV_MAX_N_IO_THREADS + 5];
|
ulint n[SRV_MAX_N_IO_THREADS + 5];
|
||||||
os_thread_id_t thread_ids[SRV_MAX_N_IO_THREADS + 5];
|
os_thread_id_t thread_ids[SRV_MAX_N_IO_THREADS + 5];
|
||||||
|
|
||||||
|
/* We use this mutex to test the return value of pthread_mutex_trylock
|
||||||
|
on successful locking. HP-UX does NOT return 0, though Linux et al do. */
|
||||||
|
os_fast_mutex_t srv_os_test_mutex;
|
||||||
|
|
||||||
|
ibool srv_os_test_mutex_is_locked = FALSE;
|
||||||
|
|
||||||
#define SRV_N_PENDING_IOS_PER_THREAD OS_AIO_N_PENDING_IOS_PER_THREAD
|
#define SRV_N_PENDING_IOS_PER_THREAD OS_AIO_N_PENDING_IOS_PER_THREAD
|
||||||
#define SRV_MAX_N_PENDING_SYNC_IOS 100
|
#define SRV_MAX_N_PENDING_SYNC_IOS 100
|
||||||
|
|
||||||
|
@ -927,6 +933,8 @@ innobase_start_or_create_for_mysql(void)
|
||||||
ulint max_arch_log_no;
|
ulint max_arch_log_no;
|
||||||
ibool start_archive;
|
ibool start_archive;
|
||||||
ulint sum_of_new_sizes;
|
ulint sum_of_new_sizes;
|
||||||
|
ulint sum_of_data_file_sizes;
|
||||||
|
ulint tablespace_size_in_header;
|
||||||
ulint err;
|
ulint err;
|
||||||
ulint i;
|
ulint i;
|
||||||
ulint k;
|
ulint k;
|
||||||
|
@ -1324,7 +1332,49 @@ innobase_start_or_create_for_mysql(void)
|
||||||
os_thread_create(&srv_master_thread, NULL, thread_ids + 1 +
|
os_thread_create(&srv_master_thread, NULL, thread_ids + 1 +
|
||||||
SRV_MAX_N_IO_THREADS);
|
SRV_MAX_N_IO_THREADS);
|
||||||
/* buf_debug_prints = TRUE; */
|
/* buf_debug_prints = TRUE; */
|
||||||
|
|
||||||
|
sum_of_data_file_sizes = 0;
|
||||||
|
|
||||||
|
for (i = 0; i < srv_n_data_files; i++) {
|
||||||
|
sum_of_data_file_sizes += srv_data_file_sizes[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
tablespace_size_in_header = fsp_header_get_tablespace_size(0);
|
||||||
|
|
||||||
|
if (!srv_auto_extend_last_data_file
|
||||||
|
&& sum_of_data_file_sizes != tablespace_size_in_header) {
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: tablespace size stored in header is %lu pages, but\n"
|
||||||
|
"InnoDB: the sum of data file sizes is %lu pages\n",
|
||||||
|
tablespace_size_in_header, sum_of_data_file_sizes);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (srv_auto_extend_last_data_file
|
||||||
|
&& sum_of_data_file_sizes < tablespace_size_in_header) {
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: tablespace size stored in header is %lu pages, but\n"
|
||||||
|
"InnoDB: the sum of data file sizes is only %lu pages\n",
|
||||||
|
tablespace_size_in_header, sum_of_data_file_sizes);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check that os_fast_mutexes work as exptected */
|
||||||
|
os_fast_mutex_init(&srv_os_test_mutex);
|
||||||
|
|
||||||
|
if (0 != os_fast_mutex_trylock(&srv_os_test_mutex)) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"InnoDB: Error: pthread_mutex_trylock returns an unexpected value on\n"
|
||||||
|
"InnoDB: success! Cannot continue.\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
os_fast_mutex_unlock(&srv_os_test_mutex);
|
||||||
|
|
||||||
|
os_fast_mutex_lock(&srv_os_test_mutex);
|
||||||
|
|
||||||
|
os_fast_mutex_unlock(&srv_os_test_mutex);
|
||||||
|
|
||||||
if (srv_print_verbose_log)
|
if (srv_print_verbose_log)
|
||||||
{
|
{
|
||||||
ut_print_timestamp(stderr);
|
ut_print_timestamp(stderr);
|
||||||
|
|
|
@ -454,7 +454,7 @@ sync_array_cell_print(
|
||||||
|
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"--Thread %lu has waited at %s line %lu for %.2f seconds the semaphore:\n",
|
"--Thread %lu has waited at %s line %lu for %.2f seconds the semaphore:\n",
|
||||||
(ulint)cell->thread, cell->file, cell->line,
|
os_thread_pf(cell->thread), cell->file, cell->line,
|
||||||
difftime(time(NULL), cell->reservation_time));
|
difftime(time(NULL), cell->reservation_time));
|
||||||
|
|
||||||
if (type == SYNC_MUTEX) {
|
if (type == SYNC_MUTEX) {
|
||||||
|
@ -486,7 +486,7 @@ sync_array_cell_print(
|
||||||
if (rwlock->writer != RW_LOCK_NOT_LOCKED) {
|
if (rwlock->writer != RW_LOCK_NOT_LOCKED) {
|
||||||
buf += sprintf(buf,
|
buf += sprintf(buf,
|
||||||
"a writer (thread id %lu) has reserved it in mode",
|
"a writer (thread id %lu) has reserved it in mode",
|
||||||
(ulint)rwlock->writer_thread);
|
os_thread_pf(rwlock->writer_thread));
|
||||||
if (rwlock->writer == RW_LOCK_EX) {
|
if (rwlock->writer == RW_LOCK_EX) {
|
||||||
buf += sprintf(buf, " exclusive\n");
|
buf += sprintf(buf, " exclusive\n");
|
||||||
} else {
|
} else {
|
||||||
|
@ -535,8 +535,8 @@ sync_array_find_thread(
|
||||||
|
|
||||||
cell = sync_array_get_nth_cell(arr, i);
|
cell = sync_array_get_nth_cell(arr, i);
|
||||||
|
|
||||||
if ((cell->wait_object != NULL)
|
if (cell->wait_object != NULL
|
||||||
&& (cell->thread == thread)) {
|
&& os_thread_eq(cell->thread, thread)) {
|
||||||
|
|
||||||
return(cell); /* Found */
|
return(cell); /* Found */
|
||||||
}
|
}
|
||||||
|
@ -651,9 +651,9 @@ sync_array_detect_deadlock(
|
||||||
sync_array_cell_print(buf, cell);
|
sync_array_cell_print(buf, cell);
|
||||||
printf(
|
printf(
|
||||||
"Mutex %lx owned by thread %lu file %s line %lu\n%s",
|
"Mutex %lx owned by thread %lu file %s line %lu\n%s",
|
||||||
(ulint)mutex, mutex->thread_id,
|
(ulint)mutex, os_thread_pf(mutex->thread_id),
|
||||||
mutex->file_name, mutex->line,
|
mutex->file_name, mutex->line, buf);
|
||||||
buf);
|
|
||||||
return(TRUE);
|
return(TRUE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -671,9 +671,9 @@ sync_array_detect_deadlock(
|
||||||
thread = debug->thread_id;
|
thread = debug->thread_id;
|
||||||
|
|
||||||
if (((debug->lock_type == RW_LOCK_EX)
|
if (((debug->lock_type == RW_LOCK_EX)
|
||||||
&& (thread != cell->thread))
|
&& !os_thread_eq(thread, cell->thread))
|
||||||
|| ((debug->lock_type == RW_LOCK_WAIT_EX)
|
|| ((debug->lock_type == RW_LOCK_WAIT_EX)
|
||||||
&& (thread != cell->thread))
|
&& !os_thread_eq(thread, cell->thread))
|
||||||
|| (debug->lock_type == RW_LOCK_SHARED)) {
|
|| (debug->lock_type == RW_LOCK_SHARED)) {
|
||||||
|
|
||||||
/* The (wait) x-lock request can block infinitely
|
/* The (wait) x-lock request can block infinitely
|
||||||
|
@ -771,7 +771,7 @@ sync_arr_cell_can_wake_up(
|
||||||
|
|
||||||
if (rw_lock_get_reader_count(lock) == 0
|
if (rw_lock_get_reader_count(lock) == 0
|
||||||
&& rw_lock_get_writer(lock) == RW_LOCK_WAIT_EX
|
&& rw_lock_get_writer(lock) == RW_LOCK_WAIT_EX
|
||||||
&& lock->writer_thread == cell->thread) {
|
&& os_thread_eq(lock->writer_thread, cell->thread)) {
|
||||||
|
|
||||||
return(TRUE);
|
return(TRUE);
|
||||||
}
|
}
|
||||||
|
@ -927,7 +927,7 @@ sync_array_print_long_waits(void)
|
||||||
&& difftime(time(NULL), cell->reservation_time) > 420) {
|
&& difftime(time(NULL), cell->reservation_time) > 420) {
|
||||||
|
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
"InnoDB: Error: semaphore wait has lasted > 420 seconds\n"
|
"InnoDB: Error: semaphore wait has lasted > 600 seconds\n"
|
||||||
"InnoDB: We intentionally crash the server, because it appears to be hung.\n"
|
"InnoDB: We intentionally crash the server, because it appears to be hung.\n"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -1011,3 +1011,4 @@ sync_array_print_info(
|
||||||
|
|
||||||
sync_array_exit(arr);
|
sync_array_exit(arr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -223,7 +223,7 @@ lock_loop:
|
||||||
if (srv_print_latch_waits) {
|
if (srv_print_latch_waits) {
|
||||||
printf(
|
printf(
|
||||||
"Thread %lu spin wait rw-s-lock at %lx cfile %s cline %lu rnds %lu\n",
|
"Thread %lu spin wait rw-s-lock at %lx cfile %s cline %lu rnds %lu\n",
|
||||||
os_thread_get_curr_id(), (ulint)lock,
|
os_thread_pf(os_thread_get_curr_id()), (ulint)lock,
|
||||||
lock->cfile_name, lock->cline, i);
|
lock->cfile_name, lock->cline, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -253,7 +253,7 @@ lock_loop:
|
||||||
if (srv_print_latch_waits) {
|
if (srv_print_latch_waits) {
|
||||||
printf(
|
printf(
|
||||||
"Thread %lu OS wait rw-s-lock at %lx cfile %s cline %lu\n",
|
"Thread %lu OS wait rw-s-lock at %lx cfile %s cline %lu\n",
|
||||||
os_thread_get_curr_id(), (ulint)lock,
|
os_thread_pf(os_thread_get_curr_id()), (ulint)lock,
|
||||||
lock->cfile_name, lock->cline);
|
lock->cfile_name, lock->cline);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -343,7 +343,8 @@ rw_lock_x_lock_low(
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if ((rw_lock_get_writer(lock) == RW_LOCK_WAIT_EX)
|
} else if ((rw_lock_get_writer(lock) == RW_LOCK_WAIT_EX)
|
||||||
&& (lock->writer_thread == os_thread_get_curr_id())) {
|
&& os_thread_eq(lock->writer_thread,
|
||||||
|
os_thread_get_curr_id())) {
|
||||||
|
|
||||||
if (rw_lock_get_reader_count(lock) == 0) {
|
if (rw_lock_get_reader_count(lock) == 0) {
|
||||||
|
|
||||||
|
@ -368,7 +369,8 @@ rw_lock_x_lock_low(
|
||||||
return(RW_LOCK_WAIT_EX);
|
return(RW_LOCK_WAIT_EX);
|
||||||
|
|
||||||
} else if ((rw_lock_get_writer(lock) == RW_LOCK_EX)
|
} else if ((rw_lock_get_writer(lock) == RW_LOCK_EX)
|
||||||
&& (lock->writer_thread == os_thread_get_curr_id())
|
&& os_thread_eq(lock->writer_thread,
|
||||||
|
os_thread_get_curr_id())
|
||||||
&& (lock->pass == 0)
|
&& (lock->pass == 0)
|
||||||
&& (pass == 0)) {
|
&& (pass == 0)) {
|
||||||
|
|
||||||
|
@ -469,7 +471,7 @@ lock_loop:
|
||||||
if (srv_print_latch_waits) {
|
if (srv_print_latch_waits) {
|
||||||
printf(
|
printf(
|
||||||
"Thread %lu spin wait rw-x-lock at %lx cfile %s cline %lu rnds %lu\n",
|
"Thread %lu spin wait rw-x-lock at %lx cfile %s cline %lu rnds %lu\n",
|
||||||
os_thread_get_curr_id(), (ulint)lock,
|
os_thread_pf(os_thread_get_curr_id()), (ulint)lock,
|
||||||
lock->cfile_name, lock->cline, i);
|
lock->cfile_name, lock->cline, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -502,8 +504,8 @@ lock_loop:
|
||||||
if (srv_print_latch_waits) {
|
if (srv_print_latch_waits) {
|
||||||
printf(
|
printf(
|
||||||
"Thread %lu OS wait for rw-x-lock at %lx cfile %s cline %lu\n",
|
"Thread %lu OS wait for rw-x-lock at %lx cfile %s cline %lu\n",
|
||||||
os_thread_get_curr_id(), (ulint)lock, lock->cfile_name,
|
os_thread_pf(os_thread_get_curr_id()), (ulint)lock,
|
||||||
lock->cline);
|
lock->cfile_name, lock->cline);
|
||||||
}
|
}
|
||||||
|
|
||||||
rw_x_system_call_count++;
|
rw_x_system_call_count++;
|
||||||
|
@ -621,7 +623,8 @@ rw_lock_remove_debug_info(
|
||||||
while (info != NULL) {
|
while (info != NULL) {
|
||||||
if ((pass == info->pass)
|
if ((pass == info->pass)
|
||||||
&& ((pass != 0)
|
&& ((pass != 0)
|
||||||
|| (info->thread_id == os_thread_get_curr_id()))
|
|| os_thread_eq(info->thread_id,
|
||||||
|
os_thread_get_curr_id()))
|
||||||
&& (info->lock_type == lock_type)) {
|
&& (info->lock_type == lock_type)) {
|
||||||
|
|
||||||
/* Found! */
|
/* Found! */
|
||||||
|
@ -676,7 +679,7 @@ rw_lock_own(
|
||||||
|
|
||||||
while (info != NULL) {
|
while (info != NULL) {
|
||||||
|
|
||||||
if ((info->thread_id == os_thread_get_curr_id())
|
if (os_thread_eq(info->thread_id, os_thread_get_curr_id())
|
||||||
&& (info->pass == 0)
|
&& (info->pass == 0)
|
||||||
&& (info->lock_type == lock_type)) {
|
&& (info->lock_type == lock_type)) {
|
||||||
|
|
||||||
|
@ -834,7 +837,7 @@ rw_lock_debug_print(
|
||||||
rwt = info->lock_type;
|
rwt = info->lock_type;
|
||||||
|
|
||||||
printf("Locked: thread %ld file %s line %ld ",
|
printf("Locked: thread %ld file %s line %ld ",
|
||||||
info->thread_id, info->file_name, info->line);
|
os_thread_pf(info->thread_id), info->file_name, info->line);
|
||||||
if (rwt == RW_LOCK_SHARED) {
|
if (rwt == RW_LOCK_SHARED) {
|
||||||
printf("S-LOCK");
|
printf("S-LOCK");
|
||||||
} else if (rwt == RW_LOCK_EX) {
|
} else if (rwt == RW_LOCK_EX) {
|
||||||
|
|
|
@ -230,7 +230,6 @@ mutex_create_func(
|
||||||
mutex->magic_n = MUTEX_MAGIC_N;
|
mutex->magic_n = MUTEX_MAGIC_N;
|
||||||
mutex->line = 0;
|
mutex->line = 0;
|
||||||
mutex->file_name = (char *) "not yet reserved";
|
mutex->file_name = (char *) "not yet reserved";
|
||||||
mutex->thread_id = ULINT_UNDEFINED;
|
|
||||||
mutex->level = SYNC_LEVEL_NONE;
|
mutex->level = SYNC_LEVEL_NONE;
|
||||||
mutex->cfile_name = cfile_name;
|
mutex->cfile_name = cfile_name;
|
||||||
mutex->cline = cline;
|
mutex->cline = cline;
|
||||||
|
@ -392,8 +391,8 @@ spin_loop:
|
||||||
if (srv_print_latch_waits) {
|
if (srv_print_latch_waits) {
|
||||||
printf(
|
printf(
|
||||||
"Thread %lu spin wait mutex at %lx cfile %s cline %lu rnds %lu\n",
|
"Thread %lu spin wait mutex at %lx cfile %s cline %lu rnds %lu\n",
|
||||||
os_thread_get_curr_id(), (ulint)mutex, mutex->cfile_name,
|
os_thread_pf(os_thread_get_curr_id()), (ulint)mutex,
|
||||||
mutex->cline, i);
|
mutex->cfile_name, mutex->cline, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_spin_round_count += i;
|
mutex_spin_round_count += i;
|
||||||
|
@ -458,7 +457,7 @@ spin_loop:
|
||||||
if (srv_print_latch_waits) {
|
if (srv_print_latch_waits) {
|
||||||
printf(
|
printf(
|
||||||
"Thread %lu spin wait succeeds at 2: mutex at %lx\n",
|
"Thread %lu spin wait succeeds at 2: mutex at %lx\n",
|
||||||
os_thread_get_curr_id(), (ulint)mutex);
|
os_thread_pf(os_thread_get_curr_id()), (ulint)mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
|
@ -476,8 +475,8 @@ spin_loop:
|
||||||
if (srv_print_latch_waits) {
|
if (srv_print_latch_waits) {
|
||||||
printf(
|
printf(
|
||||||
"Thread %lu OS wait mutex at %lx cfile %s cline %lu rnds %lu\n",
|
"Thread %lu OS wait mutex at %lx cfile %s cline %lu rnds %lu\n",
|
||||||
os_thread_get_curr_id(), (ulint)mutex, mutex->cfile_name,
|
os_thread_pf(os_thread_get_curr_id()), (ulint)mutex,
|
||||||
mutex->cline, i);
|
mutex->cfile_name, mutex->cline, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_system_call_count++;
|
mutex_system_call_count++;
|
||||||
|
@ -572,7 +571,7 @@ mutex_own(
|
||||||
return(FALSE);
|
return(FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mutex->thread_id != os_thread_get_curr_id()) {
|
if (!os_thread_eq(mutex->thread_id, os_thread_get_curr_id())) {
|
||||||
|
|
||||||
return(FALSE);
|
return(FALSE);
|
||||||
}
|
}
|
||||||
|
@ -611,7 +610,8 @@ mutex_list_print_info(void)
|
||||||
&thread_id);
|
&thread_id);
|
||||||
printf(
|
printf(
|
||||||
"Locked mutex: addr %lx thread %ld file %s line %ld\n",
|
"Locked mutex: addr %lx thread %ld file %s line %ld\n",
|
||||||
(ulint)mutex, thread_id, file_name, line);
|
(ulint)mutex, os_thread_pf(thread_id),
|
||||||
|
file_name, line);
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex = UT_LIST_GET_NEXT(list, mutex);
|
mutex = UT_LIST_GET_NEXT(list, mutex);
|
||||||
|
@ -716,7 +716,7 @@ sync_thread_level_arrays_find_slot(void)
|
||||||
|
|
||||||
slot = sync_thread_level_arrays_get_nth(i);
|
slot = sync_thread_level_arrays_get_nth(i);
|
||||||
|
|
||||||
if (slot->levels && (slot->id == id)) {
|
if (slot->levels && os_thread_eq(slot->id, id)) {
|
||||||
|
|
||||||
return(slot);
|
return(slot);
|
||||||
}
|
}
|
||||||
|
@ -780,7 +780,7 @@ sync_thread_levels_g(
|
||||||
{
|
{
|
||||||
char* file_name;
|
char* file_name;
|
||||||
ulint line;
|
ulint line;
|
||||||
ulint thread_id;
|
os_thread_id_t thread_id;
|
||||||
sync_level_t* slot;
|
sync_level_t* slot;
|
||||||
rw_lock_t* lock;
|
rw_lock_t* lock;
|
||||||
mutex_t* mutex;
|
mutex_t* mutex;
|
||||||
|
@ -810,7 +810,7 @@ sync_thread_levels_g(
|
||||||
&file_name, &line, &thread_id);
|
&file_name, &line, &thread_id);
|
||||||
|
|
||||||
printf("InnoDB: Locked mutex: addr %lx thread %ld file %s line %ld\n",
|
printf("InnoDB: Locked mutex: addr %lx thread %ld file %s line %ld\n",
|
||||||
(ulint)mutex, thread_id,
|
(ulint)mutex, os_thread_pf(thread_id),
|
||||||
file_name, line);
|
file_name, line);
|
||||||
} else {
|
} else {
|
||||||
printf("Not locked\n");
|
printf("Not locked\n");
|
||||||
|
|
|
@ -69,8 +69,8 @@ try_again:
|
||||||
|
|
||||||
local = NULL;
|
local = NULL;
|
||||||
|
|
||||||
HASH_SEARCH(hash, thr_local_hash, os_thread_conv_id_to_ulint(id),
|
HASH_SEARCH(hash, thr_local_hash, os_thread_pf(id),
|
||||||
local, local->id == id);
|
local, os_thread_eq(local->id, id));
|
||||||
if (local == NULL) {
|
if (local == NULL) {
|
||||||
mutex_exit(&thr_local_mutex);
|
mutex_exit(&thr_local_mutex);
|
||||||
|
|
||||||
|
@ -173,7 +173,7 @@ thr_local_create(void)
|
||||||
mutex_enter(&thr_local_mutex);
|
mutex_enter(&thr_local_mutex);
|
||||||
|
|
||||||
HASH_INSERT(thr_local_t, hash, thr_local_hash,
|
HASH_INSERT(thr_local_t, hash, thr_local_hash,
|
||||||
os_thread_conv_id_to_ulint(os_thread_get_curr_id()),
|
os_thread_pf(os_thread_get_curr_id()),
|
||||||
local);
|
local);
|
||||||
|
|
||||||
mutex_exit(&thr_local_mutex);
|
mutex_exit(&thr_local_mutex);
|
||||||
|
@ -193,8 +193,8 @@ thr_local_free(
|
||||||
|
|
||||||
/* Look for the local struct in the hash table */
|
/* Look for the local struct in the hash table */
|
||||||
|
|
||||||
HASH_SEARCH(hash, thr_local_hash, os_thread_conv_id_to_ulint(id),
|
HASH_SEARCH(hash, thr_local_hash, os_thread_pf(id),
|
||||||
local, local->id == id);
|
local, os_thread_eq(local->id, id));
|
||||||
if (local == NULL) {
|
if (local == NULL) {
|
||||||
mutex_exit(&thr_local_mutex);
|
mutex_exit(&thr_local_mutex);
|
||||||
|
|
||||||
|
@ -202,7 +202,7 @@ thr_local_free(
|
||||||
}
|
}
|
||||||
|
|
||||||
HASH_DELETE(thr_local_t, hash, thr_local_hash,
|
HASH_DELETE(thr_local_t, hash, thr_local_hash,
|
||||||
os_thread_conv_id_to_ulint(id), local);
|
os_thread_pf(id), local);
|
||||||
|
|
||||||
mutex_exit(&thr_local_mutex);
|
mutex_exit(&thr_local_mutex);
|
||||||
|
|
||||||
|
|
|
@ -493,6 +493,34 @@ trx_sys_update_mysql_binlog_offset(
|
||||||
MLOG_4BYTES, mtr);
|
MLOG_4BYTES, mtr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*********************************************************************
|
||||||
|
Prints to stdout the MySQL binlog info in the system header if the
|
||||||
|
magic number shows it valid. */
|
||||||
|
|
||||||
|
void
|
||||||
|
trx_sys_print_mysql_binlog_offset_from_page(
|
||||||
|
/*========================================*/
|
||||||
|
byte* page) /* in: buffer containing the trx system header page,
|
||||||
|
i.e., page number TRX_SYS_PAGE_NO in the tablespace */
|
||||||
|
{
|
||||||
|
trx_sysf_t* sys_header;
|
||||||
|
|
||||||
|
sys_header = page + TRX_SYS;
|
||||||
|
|
||||||
|
if (mach_read_from_4(sys_header + TRX_SYS_MYSQL_LOG_INFO
|
||||||
|
+ TRX_SYS_MYSQL_LOG_MAGIC_N_FLD)
|
||||||
|
== TRX_SYS_MYSQL_LOG_MAGIC_N) {
|
||||||
|
|
||||||
|
printf(
|
||||||
|
"ibbackup: Last MySQL binlog file position %lu %lu, file name %s\n",
|
||||||
|
mach_read_from_4(sys_header + TRX_SYS_MYSQL_LOG_INFO
|
||||||
|
+ TRX_SYS_MYSQL_LOG_OFFSET_HIGH),
|
||||||
|
mach_read_from_4(sys_header + TRX_SYS_MYSQL_LOG_INFO
|
||||||
|
+ TRX_SYS_MYSQL_LOG_OFFSET_LOW),
|
||||||
|
sys_header + TRX_SYS_MYSQL_LOG_INFO + TRX_SYS_MYSQL_LOG_NAME);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
Prints to stderr the MySQL binlog offset info in the trx system header if
|
Prints to stderr the MySQL binlog offset info in the trx system header if
|
||||||
the magic number shows it valid. */
|
the magic number shows it valid. */
|
||||||
|
|
|
@ -72,6 +72,7 @@ trx_create(
|
||||||
|
|
||||||
trx->type = TRX_USER;
|
trx->type = TRX_USER;
|
||||||
trx->conc_state = TRX_NOT_STARTED;
|
trx->conc_state = TRX_NOT_STARTED;
|
||||||
|
trx->start_time = time(NULL);
|
||||||
|
|
||||||
trx->check_foreigns = TRUE;
|
trx->check_foreigns = TRUE;
|
||||||
trx->check_unique_secondary = TRUE;
|
trx->check_unique_secondary = TRUE;
|
||||||
|
@ -516,6 +517,7 @@ trx_start_low(
|
||||||
if (trx->type == TRX_PURGE) {
|
if (trx->type == TRX_PURGE) {
|
||||||
trx->id = ut_dulint_zero;
|
trx->id = ut_dulint_zero;
|
||||||
trx->conc_state = TRX_ACTIVE;
|
trx->conc_state = TRX_ACTIVE;
|
||||||
|
trx->start_time = time(NULL);
|
||||||
|
|
||||||
return(TRUE);
|
return(TRUE);
|
||||||
}
|
}
|
||||||
|
@ -539,6 +541,7 @@ trx_start_low(
|
||||||
trx->rseg = rseg;
|
trx->rseg = rseg;
|
||||||
|
|
||||||
trx->conc_state = TRX_ACTIVE;
|
trx->conc_state = TRX_ACTIVE;
|
||||||
|
trx->start_time = time(NULL);
|
||||||
|
|
||||||
UT_LIST_ADD_FIRST(trx_list, trx_sys->trx_list, trx);
|
UT_LIST_ADD_FIRST(trx_list, trx_sys->trx_list, trx);
|
||||||
|
|
||||||
|
@ -1465,10 +1468,26 @@ trx_print(
|
||||||
500 bytes */
|
500 bytes */
|
||||||
trx_t* trx) /* in: transaction */
|
trx_t* trx) /* in: transaction */
|
||||||
{
|
{
|
||||||
buf += sprintf(buf, "TRANSACTION %lu %lu, OS thread id %lu",
|
char* start_of_line;
|
||||||
|
|
||||||
|
buf += sprintf(buf, "TRANSACTION %lu %lu",
|
||||||
ut_dulint_get_high(trx->id),
|
ut_dulint_get_high(trx->id),
|
||||||
ut_dulint_get_low(trx->id),
|
ut_dulint_get_low(trx->id));
|
||||||
(ulint)trx->mysql_thread_id);
|
|
||||||
|
switch (trx->conc_state) {
|
||||||
|
case TRX_NOT_STARTED: buf += sprintf(buf,
|
||||||
|
", not started"); break;
|
||||||
|
case TRX_ACTIVE: buf += sprintf(buf,
|
||||||
|
", ACTIVE %lu sec",
|
||||||
|
(ulint)difftime(time(NULL), trx->start_time)); break;
|
||||||
|
case TRX_COMMITTED_IN_MEMORY: buf += sprintf(buf,
|
||||||
|
", COMMITTED IN MEMORY");
|
||||||
|
break;
|
||||||
|
default: buf += sprintf(buf, " state %lu", trx->conc_state);
|
||||||
|
}
|
||||||
|
|
||||||
|
buf += sprintf(buf, ", OS thread id %lu",
|
||||||
|
os_thread_pf(trx->mysql_thread_id));
|
||||||
|
|
||||||
if (ut_strlen(trx->op_info) > 0) {
|
if (ut_strlen(trx->op_info) > 0) {
|
||||||
buf += sprintf(buf, " %s", trx->op_info);
|
buf += sprintf(buf, " %s", trx->op_info);
|
||||||
|
@ -1477,33 +1496,29 @@ trx_print(
|
||||||
if (trx->type != TRX_USER) {
|
if (trx->type != TRX_USER) {
|
||||||
buf += sprintf(buf, " purge trx");
|
buf += sprintf(buf, " purge trx");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
buf += sprintf(buf, "\n");
|
||||||
|
|
||||||
switch (trx->conc_state) {
|
start_of_line = buf;
|
||||||
case TRX_NOT_STARTED: buf += sprintf(buf,
|
|
||||||
", not started"); break;
|
|
||||||
case TRX_ACTIVE: buf += sprintf(buf,
|
|
||||||
", active"); break;
|
|
||||||
case TRX_COMMITTED_IN_MEMORY: buf += sprintf(buf,
|
|
||||||
", committed in memory");
|
|
||||||
break;
|
|
||||||
default: buf += sprintf(buf, " state %lu", trx->conc_state);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (trx->que_state) {
|
switch (trx->que_state) {
|
||||||
case TRX_QUE_RUNNING: buf += sprintf(buf,
|
case TRX_QUE_RUNNING: break;
|
||||||
", runs or sleeps"); break;
|
|
||||||
case TRX_QUE_LOCK_WAIT: buf += sprintf(buf,
|
case TRX_QUE_LOCK_WAIT: buf += sprintf(buf,
|
||||||
", lock wait"); break;
|
"LOCK WAIT "); break;
|
||||||
case TRX_QUE_ROLLING_BACK: buf += sprintf(buf,
|
case TRX_QUE_ROLLING_BACK: buf += sprintf(buf,
|
||||||
", rolling back"); break;
|
"ROLLING BACK "); break;
|
||||||
case TRX_QUE_COMMITTING: buf += sprintf(buf,
|
case TRX_QUE_COMMITTING: buf += sprintf(buf,
|
||||||
", committing"); break;
|
"COMMITTING "); break;
|
||||||
default: buf += sprintf(buf, " que state %lu", trx->que_state);
|
default: buf += sprintf(buf, "que state %lu", trx->que_state);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (0 < UT_LIST_GET_LEN(trx->trx_locks)) {
|
if (0 < UT_LIST_GET_LEN(trx->trx_locks) ||
|
||||||
buf += sprintf(buf, ", has %lu lock struct(s)",
|
mem_heap_get_size(trx->lock_heap) > 400) {
|
||||||
UT_LIST_GET_LEN(trx->trx_locks));
|
|
||||||
|
buf += sprintf(buf,
|
||||||
|
"%lu lock struct(s), heap size %lu",
|
||||||
|
UT_LIST_GET_LEN(trx->trx_locks),
|
||||||
|
mem_heap_get_size(trx->lock_heap));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (trx->has_search_latch) {
|
if (trx->has_search_latch) {
|
||||||
|
@ -1515,7 +1530,10 @@ trx_print(
|
||||||
ut_dulint_get_low(trx->undo_no));
|
ut_dulint_get_low(trx->undo_no));
|
||||||
}
|
}
|
||||||
|
|
||||||
buf += sprintf(buf, "\n");
|
if (buf != start_of_line) {
|
||||||
|
|
||||||
|
buf += sprintf(buf, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
if (trx->mysql_thd != NULL) {
|
if (trx->mysql_thd != NULL) {
|
||||||
innobase_mysql_print_thd(buf, trx->mysql_thd);
|
innobase_mysql_print_thd(buf, trx->mysql_thd);
|
||||||
|
|
|
@ -49,6 +49,10 @@ link_sources:
|
||||||
rm -f $(srcdir)/$$f; \
|
rm -f $(srcdir)/$$f; \
|
||||||
@LN_CP_F@ $(srcdir)/../strings/$$f $(srcdir)/$$f; \
|
@LN_CP_F@ $(srcdir)/../strings/$$f $(srcdir)/$$f; \
|
||||||
done; \
|
done; \
|
||||||
|
for f in $(mystringsgen); do \
|
||||||
|
rm -f $(srcdir)/$$f; \
|
||||||
|
@LN_CP_F@ ../strings/$$f $(srcdir)/$$f; \
|
||||||
|
done; \
|
||||||
for f in $$qs; do \
|
for f in $$qs; do \
|
||||||
rm -f $(srcdir)/$$f; \
|
rm -f $(srcdir)/$$f; \
|
||||||
@LN_CP_F@ $(srcdir)/../sql/$$f $(srcdir)/$$f; \
|
@LN_CP_F@ $(srcdir)/../sql/$$f $(srcdir)/$$f; \
|
||||||
|
|
|
@ -42,7 +42,8 @@ mystringsobjects = strmov.lo strxmov.lo strxnmov.lo strnmov.lo \
|
||||||
bchange.lo bmove.lo bmove_upp.lo longlong2str.lo \
|
bchange.lo bmove.lo bmove_upp.lo longlong2str.lo \
|
||||||
strtoull.lo strtoll.lo llstr.lo \
|
strtoull.lo strtoll.lo llstr.lo \
|
||||||
ctype.lo $(LTCHARSET_OBJS)
|
ctype.lo $(LTCHARSET_OBJS)
|
||||||
mystringsextra= strto.c ctype_autoconf.c
|
mystringsextra= strto.c
|
||||||
|
mystringsgen= ctype_autoconf.c
|
||||||
dbugobjects = dbug.lo # IT IS IN SAFEMALLOC.C sanity.lo
|
dbugobjects = dbug.lo # IT IS IN SAFEMALLOC.C sanity.lo
|
||||||
mysysheaders = mysys_priv.h my_static.h
|
mysysheaders = mysys_priv.h my_static.h
|
||||||
mysysobjects1 = my_init.lo my_static.lo my_malloc.lo my_realloc.lo \
|
mysysobjects1 = my_init.lo my_static.lo my_malloc.lo my_realloc.lo \
|
||||||
|
@ -83,9 +84,8 @@ clean-local:
|
||||||
rm -f `echo $(mystringsobjects) | sed "s;\.lo;.c;g"` \
|
rm -f `echo $(mystringsobjects) | sed "s;\.lo;.c;g"` \
|
||||||
`echo $(dbugobjects) | sed "s;\.lo;.c;g"` \
|
`echo $(dbugobjects) | sed "s;\.lo;.c;g"` \
|
||||||
`echo $(mysysobjects) | sed "s;\.lo;.c;g"` \
|
`echo $(mysysobjects) | sed "s;\.lo;.c;g"` \
|
||||||
`echo $(vio_objects) | sed "s;\.lo;.c;g"` \
|
$(mystringsextra) $(mystringsgen) $(mysysheaders) \
|
||||||
$(mystringsextra) $(mysysheaders) ctype_extra_sources.c \
|
ctype_extra_sources.c net.c ../linked_client_sources
|
||||||
net.c ../linked_client_sources
|
|
||||||
|
|
||||||
ctype_extra_sources.c: conf_to_src
|
ctype_extra_sources.c: conf_to_src
|
||||||
./conf_to_src $(top_srcdir) @CHARSETS_NEED_SOURCE@ > \
|
./conf_to_src $(top_srcdir) @CHARSETS_NEED_SOURCE@ > \
|
||||||
|
|
|
@ -151,7 +151,7 @@ int vio_write(Vio * vio, const gptr buf, int size)
|
||||||
DBUG_RETURN(size);
|
DBUG_RETURN(size);
|
||||||
}
|
}
|
||||||
|
|
||||||
int vio_blocking(Vio * vio, my_bool set_blocking_mode)
|
int vio_blocking(Vio * vio, my_bool set_blocking_mode, my_bool *old_mode)
|
||||||
{
|
{
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
|
@ -395,3 +395,25 @@ gender dist_count percentage
|
||||||
M 1 20.00
|
M 1 20.00
|
||||||
F 3 60.00
|
F 3 60.00
|
||||||
drop table t1,t2;
|
drop table t1,t2;
|
||||||
|
CREATE TABLE t1 (ID1 int, ID2 int, ID int NOT NULL AUTO_INCREMENT,PRIMARY KEY(ID
|
||||||
|
));
|
||||||
|
insert into t1 values (1,244,NULL),(2,243,NULL),(134,223,NULL),(185,186,NULL);
|
||||||
|
select S.ID as xID, S.ID1 as xID1 from t1 as S left join t1 as yS on S.ID1 between yS.ID1 and yS.ID2;
|
||||||
|
xID xID1
|
||||||
|
1 1
|
||||||
|
2 2
|
||||||
|
2 2
|
||||||
|
3 134
|
||||||
|
3 134
|
||||||
|
3 134
|
||||||
|
4 185
|
||||||
|
4 185
|
||||||
|
4 185
|
||||||
|
4 185
|
||||||
|
select S.ID as xID, S.ID1 as xID1, repeat('*',count(distinct yS.ID)) as Level from t1 as S left join t1 as yS on S.ID1 between yS.ID1 and yS.ID2 group by xID order by xID1;
|
||||||
|
xID xID1 Level
|
||||||
|
1 1 *
|
||||||
|
2 2 **
|
||||||
|
3 134 ***
|
||||||
|
4 185 ****
|
||||||
|
drop table t1;
|
||||||
|
|
|
@ -301,3 +301,14 @@ insert into t2 values (1, '2002-06-09'),(2, '2002-06-09'),(1, '2002-06-09'),(3,
|
||||||
select u.gender as gender, count(distinct u.id) as dist_count, (count(distinct u.id)/5*100) as percentage from t1 u, t2 l where l.user_id = u.id group by u.gender;
|
select u.gender as gender, count(distinct u.id) as dist_count, (count(distinct u.id)/5*100) as percentage from t1 u, t2 l where l.user_id = u.id group by u.gender;
|
||||||
select u.gender as gender, count(distinct u.id) as dist_count, (count(distinct u.id)/5*100) as percentage from t1 u, t2 l where l.user_id = u.id group by u.gender order by percentage;
|
select u.gender as gender, count(distinct u.id) as dist_count, (count(distinct u.id)/5*100) as percentage from t1 u, t2 l where l.user_id = u.id group by u.gender order by percentage;
|
||||||
drop table t1,t2;
|
drop table t1,t2;
|
||||||
|
|
||||||
|
#
|
||||||
|
# The GROUP BY returned rows in wrong order in 3.23.51
|
||||||
|
#
|
||||||
|
|
||||||
|
CREATE TABLE t1 (ID1 int, ID2 int, ID int NOT NULL AUTO_INCREMENT,PRIMARY KEY(ID
|
||||||
|
));
|
||||||
|
insert into t1 values (1,244,NULL),(2,243,NULL),(134,223,NULL),(185,186,NULL);
|
||||||
|
select S.ID as xID, S.ID1 as xID1 from t1 as S left join t1 as yS on S.ID1 between yS.ID1 and yS.ID2;
|
||||||
|
select S.ID as xID, S.ID1 as xID1, repeat('*',count(distinct yS.ID)) as Level from t1 as S left join t1 as yS on S.ID1 between yS.ID1 and yS.ID2 group by xID order by xID1;
|
||||||
|
drop table t1;
|
||||||
|
|
|
@ -438,6 +438,12 @@ int my_pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex,
|
||||||
|
|
||||||
|
|
||||||
#ifdef HPUX
|
#ifdef HPUX
|
||||||
|
/*
|
||||||
|
In HP-UX-10.20 and other old Posix 1003.4a Draft 4 implementations
|
||||||
|
pthread_mutex_trylock returns 1 on success, not 0 like
|
||||||
|
pthread_mutex_lock
|
||||||
|
*/
|
||||||
|
|
||||||
int my_pthread_mutex_trylock(pthread_mutex_t *mutex)
|
int my_pthread_mutex_trylock(pthread_mutex_t *mutex)
|
||||||
{
|
{
|
||||||
int error=pthread_mutex_trylock(mutex);
|
int error=pthread_mutex_trylock(mutex);
|
||||||
|
@ -447,9 +453,9 @@ int my_pthread_mutex_trylock(pthread_mutex_t *mutex)
|
||||||
error=errno;
|
error=errno;
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
/* Some help functions */
|
/* Some help functions */
|
||||||
|
|
||||||
int pthread_no_free(void *not_used __attribute__((unused)))
|
int pthread_no_free(void *not_used __attribute__((unused)))
|
||||||
|
|
|
@ -21,6 +21,7 @@
|
||||||
|
|
||||||
#include <my_global.h>
|
#include <my_global.h>
|
||||||
#include <my_semaphore.h>
|
#include <my_semaphore.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
#if !defined(__WIN__) && !defined(HAVE_SEMAPHORE_H)
|
#if !defined(__WIN__) && !defined(HAVE_SEMAPHORE_H)
|
||||||
|
|
||||||
|
|
|
@ -50,7 +50,8 @@ Usage: $0 db_name[./table_regex/] [new_db_name | directory]
|
||||||
-?, --help display this helpscreen and exit
|
-?, --help display this helpscreen and exit
|
||||||
-u, --user=# user for database login if not current user
|
-u, --user=# user for database login if not current user
|
||||||
-p, --password=# password to use when connecting to server
|
-p, --password=# password to use when connecting to server
|
||||||
-P, --port=# port to use when connecting to local server
|
-h, --host=# Hostname for local server when connecting over TCP/IP
|
||||||
|
-P, --port=# port to use when connecting to local server with TCP/IP
|
||||||
-S, --socket=# socket to use when connecting to local server
|
-S, --socket=# socket to use when connecting to local server
|
||||||
|
|
||||||
--allowold don\'t abort if target already exists (rename it _old)
|
--allowold don\'t abort if target already exists (rename it _old)
|
||||||
|
@ -155,8 +156,8 @@ $opt{quiet} = 0 if $opt{debug};
|
||||||
$opt{allowold} = 1 if $opt{keepold};
|
$opt{allowold} = 1 if $opt{keepold};
|
||||||
|
|
||||||
# --- connect to the database ---
|
# --- connect to the database ---
|
||||||
my $dsn = ";host=localhost";
|
my $dsn;
|
||||||
$dsn = ";host=127.0.0.1" if $opt{port}; # use TCP/IP if port was given
|
$dsn = ";host=" . (defined($opt{host}) ? $opt{host} : "localhost");
|
||||||
$dsn .= ";port=$opt{port}" if $opt{port};
|
$dsn .= ";port=$opt{port}" if $opt{port};
|
||||||
$dsn .= ";mysql_socket=$opt{socket}" if $opt{socket};
|
$dsn .= ";mysql_socket=$opt{socket}" if $opt{socket};
|
||||||
|
|
||||||
|
@ -891,9 +892,15 @@ user for database login if not current user
|
||||||
|
|
||||||
password to use when connecting to server
|
password to use when connecting to server
|
||||||
|
|
||||||
|
=item -h, -h, --host=#
|
||||||
|
|
||||||
|
Hostname for local server when connecting over TCP/IP. By specifying this
|
||||||
|
different from 'localhost' will trigger mysqlhotcopy to use TCP/IP connection.
|
||||||
|
|
||||||
=item -P, --port=#
|
=item -P, --port=#
|
||||||
|
|
||||||
port to use when connecting to local server
|
port to use when connecting to MySQL server with TCP/IP. This is only used
|
||||||
|
when using the --host option.
|
||||||
|
|
||||||
=item -S, --socket=#
|
=item -S, --socket=#
|
||||||
|
|
||||||
|
|
371
sql/ha_innodb.cc
371
sql/ha_innodb.cc
|
@ -1,4 +1,4 @@
|
||||||
/* Copyright (C) 2000 MySQL AB & InnoDB Oy
|
/* Copyright (C) 2000 MySQL AB & Innobase Oy
|
||||||
|
|
||||||
This program is free software; you can redistribute it and/or modify
|
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
|
it under the terms of the GNU General Public License as published by
|
||||||
|
@ -55,6 +55,7 @@ typedef byte mysql_byte;
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#include "../innobase/include/univ.i"
|
#include "../innobase/include/univ.i"
|
||||||
#include "../innobase/include/os0file.h"
|
#include "../innobase/include/os0file.h"
|
||||||
|
#include "../innobase/include/os0thread.h"
|
||||||
#include "../innobase/include/srv0start.h"
|
#include "../innobase/include/srv0start.h"
|
||||||
#include "../innobase/include/srv0srv.h"
|
#include "../innobase/include/srv0srv.h"
|
||||||
#include "../innobase/include/trx0roll.h"
|
#include "../innobase/include/trx0roll.h"
|
||||||
|
@ -110,8 +111,6 @@ my_bool innobase_fast_shutdown = TRUE;
|
||||||
specify any startup options.
|
specify any startup options.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* innobase_data_file_path=ibdata:15,idata2:1,... */
|
|
||||||
|
|
||||||
char *innobase_data_file_path= (char*) "ibdata1:10M:autoextend";
|
char *innobase_data_file_path= (char*) "ibdata1:10M:autoextend";
|
||||||
static char *internal_innobase_data_file_path=0;
|
static char *internal_innobase_data_file_path=0;
|
||||||
|
|
||||||
|
@ -138,8 +137,9 @@ static void innobase_print_error(const char* db_errpfx, char* buffer);
|
||||||
/* General functions */
|
/* General functions */
|
||||||
|
|
||||||
/**********************************************************************
|
/**********************************************************************
|
||||||
Releases possible search latch, auto inc lock, and InnoDB thread FIFO ticket.
|
Releases possible search latch and InnoDB thread FIFO ticket. These should
|
||||||
These should be released at each SQL statement end. */
|
be released at each SQL statement end. It does no harm to release these
|
||||||
|
also in the middle of an SQL statement. */
|
||||||
static
|
static
|
||||||
void
|
void
|
||||||
innobase_release_stat_resources(
|
innobase_release_stat_resources(
|
||||||
|
@ -150,16 +150,6 @@ innobase_release_stat_resources(
|
||||||
trx_search_latch_release_if_reserved(trx);
|
trx_search_latch_release_if_reserved(trx);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (trx->auto_inc_lock) {
|
|
||||||
|
|
||||||
/* If we had reserved the auto-inc lock for
|
|
||||||
some table in this SQL statement, we release it now */
|
|
||||||
|
|
||||||
srv_conc_enter_innodb(trx);
|
|
||||||
row_unlock_table_autoinc_for_mysql(trx);
|
|
||||||
srv_conc_exit_innodb(trx);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (trx->declared_to_be_inside_innodb) {
|
if (trx->declared_to_be_inside_innodb) {
|
||||||
/* Release our possible ticket in the FIFO */
|
/* Release our possible ticket in the FIFO */
|
||||||
|
|
||||||
|
@ -646,6 +636,16 @@ innobase_commit(
|
||||||
|
|
||||||
trx = check_trx_exists(thd);
|
trx = check_trx_exists(thd);
|
||||||
|
|
||||||
|
if (trx->auto_inc_lock) {
|
||||||
|
|
||||||
|
/* If we had reserved the auto-inc lock for
|
||||||
|
some table in this SQL statement, we release it now */
|
||||||
|
|
||||||
|
srv_conc_enter_innodb(trx);
|
||||||
|
row_unlock_table_autoinc_for_mysql(trx);
|
||||||
|
srv_conc_exit_innodb(trx);
|
||||||
|
}
|
||||||
|
|
||||||
if (trx_handle != (void*)&innodb_dummy_stmt_trx_handle) {
|
if (trx_handle != (void*)&innodb_dummy_stmt_trx_handle) {
|
||||||
innobase_commit_low(trx);
|
innobase_commit_low(trx);
|
||||||
}
|
}
|
||||||
|
@ -714,6 +714,16 @@ innobase_rollback(
|
||||||
|
|
||||||
trx = check_trx_exists(thd);
|
trx = check_trx_exists(thd);
|
||||||
|
|
||||||
|
if (trx->auto_inc_lock) {
|
||||||
|
|
||||||
|
/* If we had reserved the auto-inc lock for
|
||||||
|
some table in this SQL statement, we release it now */
|
||||||
|
|
||||||
|
srv_conc_enter_innodb(trx);
|
||||||
|
row_unlock_table_autoinc_for_mysql(trx);
|
||||||
|
srv_conc_exit_innodb(trx);
|
||||||
|
}
|
||||||
|
|
||||||
srv_conc_enter_innodb(trx);
|
srv_conc_enter_innodb(trx);
|
||||||
|
|
||||||
if (trx_handle != (void*)&innodb_dummy_stmt_trx_handle) {
|
if (trx_handle != (void*)&innodb_dummy_stmt_trx_handle) {
|
||||||
|
@ -842,7 +852,7 @@ normalize_table_name(
|
||||||
}
|
}
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
Creates and opens a handle to a table which already exists in an Innobase
|
Creates and opens a handle to a table which already exists in an InnoDB
|
||||||
database. */
|
database. */
|
||||||
|
|
||||||
int
|
int
|
||||||
|
@ -913,13 +923,13 @@ have moved .frm files to another database?",
|
||||||
|
|
||||||
primary_key = MAX_KEY;
|
primary_key = MAX_KEY;
|
||||||
|
|
||||||
if (!row_table_got_default_clust_index(ib_table)) {
|
/* Allocate a buffer for a 'row reference'. A row reference is
|
||||||
|
a string of bytes of length ref_length which uniquely specifies
|
||||||
|
a row in our table. Note that MySQL may also compare two row
|
||||||
|
references for equality by doing a simple memcmp on the strings
|
||||||
|
of length ref_length! */
|
||||||
|
|
||||||
/* If we automatically created the clustered index,
|
if (!row_table_got_default_clust_index(ib_table)) {
|
||||||
then MySQL does not know about it and it must not be aware
|
|
||||||
of the index used on scan, to avoid checking if we update
|
|
||||||
the column of the index. The column is the row id in
|
|
||||||
the automatical case, and it will not be updated. */
|
|
||||||
|
|
||||||
((row_prebuilt_t*)innobase_prebuilt)
|
((row_prebuilt_t*)innobase_prebuilt)
|
||||||
->clust_index_was_generated = FALSE;
|
->clust_index_was_generated = FALSE;
|
||||||
|
@ -928,13 +938,13 @@ have moved .frm files to another database?",
|
||||||
key_used_on_scan = 0;
|
key_used_on_scan = 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
MySQL allocates the buffer for ref.
|
MySQL allocates the buffer for ref. key_info->key_length
|
||||||
This includes all keys + one byte for each column
|
includes space for all key columns + one byte for each column
|
||||||
that may be NULL.
|
that may be NULL. ref_length must be as exact as possible to
|
||||||
The ref_length must be exact as possible as
|
save space, because all row reference buffers are allocated
|
||||||
all reference buffers are allocated based on this.
|
based on ref_length.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
ref_length = table->key_info->key_length;
|
ref_length = table->key_info->key_length;
|
||||||
} else {
|
} else {
|
||||||
((row_prebuilt_t*)innobase_prebuilt)
|
((row_prebuilt_t*)innobase_prebuilt)
|
||||||
|
@ -942,6 +952,15 @@ have moved .frm files to another database?",
|
||||||
|
|
||||||
ref_length = DATA_ROW_ID_LEN;
|
ref_length = DATA_ROW_ID_LEN;
|
||||||
|
|
||||||
|
/*
|
||||||
|
If we automatically created the clustered index, then
|
||||||
|
MySQL does not know about it, and MySQL must NOT be aware
|
||||||
|
of the index used on scan, to make it avoid checking if we
|
||||||
|
update the column of the index. That is why we assert below
|
||||||
|
that key_used_on_scan is the undefined value MAX_KEY.
|
||||||
|
The column is the row id in the automatical generation case,
|
||||||
|
and it will never be updated anyway.
|
||||||
|
*/
|
||||||
DBUG_ASSERT(key_used_on_scan == MAX_KEY);
|
DBUG_ASSERT(key_used_on_scan == MAX_KEY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1188,7 +1207,8 @@ get_innobase_type_from_mysql_type(
|
||||||
}
|
}
|
||||||
|
|
||||||
/***********************************************************************
|
/***********************************************************************
|
||||||
Stores a key value for a row to a buffer. */
|
Stores a key value for a row to a buffer. This must currently only be used
|
||||||
|
to store a row reference to the 'ref' buffer of this table handle! */
|
||||||
|
|
||||||
uint
|
uint
|
||||||
ha_innobase::store_key_val_for_row(
|
ha_innobase::store_key_val_for_row(
|
||||||
|
@ -1196,7 +1216,8 @@ ha_innobase::store_key_val_for_row(
|
||||||
/* out: key value length as stored in buff */
|
/* out: key value length as stored in buff */
|
||||||
uint keynr, /* in: key number */
|
uint keynr, /* in: key number */
|
||||||
char* buff, /* in/out: buffer for the key value (in MySQL
|
char* buff, /* in/out: buffer for the key value (in MySQL
|
||||||
format) */
|
format); currently this MUST be the 'ref'
|
||||||
|
buffer! */
|
||||||
const mysql_byte* record)/* in: row in MySQL format */
|
const mysql_byte* record)/* in: row in MySQL format */
|
||||||
{
|
{
|
||||||
KEY* key_info = table->key_info + keynr;
|
KEY* key_info = table->key_info + keynr;
|
||||||
|
@ -1225,8 +1246,9 @@ ha_innobase::store_key_val_for_row(
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
We have to zero-fill the buffer to be able to compare two
|
We have to zero-fill the 'ref' buffer so that MySQL is able to
|
||||||
keys to see if they are equal
|
use a simple memcmp to compare two key values to determine if they
|
||||||
|
are equal
|
||||||
*/
|
*/
|
||||||
bzero(buff, (ref_length- (uint) (buff - buff_start)));
|
bzero(buff, (ref_length- (uint) (buff - buff_start)));
|
||||||
DBUG_RETURN(ref_length);
|
DBUG_RETURN(ref_length);
|
||||||
|
@ -1404,6 +1426,7 @@ ha_innobase::write_row(
|
||||||
row_prebuilt_t* prebuilt = (row_prebuilt_t*)innobase_prebuilt;
|
row_prebuilt_t* prebuilt = (row_prebuilt_t*)innobase_prebuilt;
|
||||||
int error;
|
int error;
|
||||||
longlong auto_inc;
|
longlong auto_inc;
|
||||||
|
longlong dummy;
|
||||||
|
|
||||||
DBUG_ENTER("ha_innobase::write_row");
|
DBUG_ENTER("ha_innobase::write_row");
|
||||||
|
|
||||||
|
@ -1426,7 +1449,31 @@ ha_innobase::write_row(
|
||||||
if (table->next_number_field && record == table->record[0]) {
|
if (table->next_number_field && record == table->record[0]) {
|
||||||
/* This is the case where the table has an
|
/* This is the case where the table has an
|
||||||
auto-increment column */
|
auto-increment column */
|
||||||
|
|
||||||
|
/* Initialize the auto-inc counter if it has not been
|
||||||
|
initialized yet */
|
||||||
|
|
||||||
|
if (0 == dict_table_autoinc_peek(prebuilt->table)) {
|
||||||
|
|
||||||
|
/* This call initializes the counter */
|
||||||
|
error = innobase_read_and_init_auto_inc(&dummy);
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
/* Deadlock or lock wait timeout */
|
||||||
|
|
||||||
|
goto func_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We have to set sql_stat_start to TRUE because
|
||||||
|
the above call probably has called a select, and
|
||||||
|
has reset that flag; row_insert_for_mysql has to
|
||||||
|
know to set the IX intention lock on the table,
|
||||||
|
something it only does at the start of each
|
||||||
|
statement */
|
||||||
|
|
||||||
|
prebuilt->sql_stat_start = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
/* Fetch the value the user possibly has set in the
|
/* Fetch the value the user possibly has set in the
|
||||||
autoincrement field */
|
autoincrement field */
|
||||||
|
|
||||||
|
@ -1459,10 +1506,9 @@ ha_innobase::write_row(
|
||||||
}
|
}
|
||||||
|
|
||||||
if (auto_inc != 0) {
|
if (auto_inc != 0) {
|
||||||
/* This call will calculate the max of the
|
/* This call will calculate the max of the current
|
||||||
current value and the value supplied by the user, if
|
value and the value supplied by the user and
|
||||||
the auto_inc counter is already initialized
|
update the counter accordingly */
|
||||||
for the table */
|
|
||||||
|
|
||||||
/* We have to use the transactional lock mechanism
|
/* We have to use the transactional lock mechanism
|
||||||
on the auto-inc counter of the table to ensure
|
on the auto-inc counter of the table to ensure
|
||||||
|
@ -1502,46 +1548,18 @@ ha_innobase::write_row(
|
||||||
auto_inc = dict_table_autoinc_get(prebuilt->table);
|
auto_inc = dict_table_autoinc_get(prebuilt->table);
|
||||||
srv_conc_exit_innodb(prebuilt->trx);
|
srv_conc_exit_innodb(prebuilt->trx);
|
||||||
|
|
||||||
/* If auto_inc is now != 0 the autoinc counter
|
/* We can give the new value for MySQL to place in
|
||||||
was already initialized for the table: we can give
|
the field */
|
||||||
the new value for MySQL to place in the field */
|
|
||||||
|
|
||||||
if (auto_inc != 0) {
|
user_thd->next_insert_id = auto_inc;
|
||||||
user_thd->next_insert_id = auto_inc;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* This call of a handler.cc function places
|
||||||
|
user_thd->next_insert_id to the column value, if the column
|
||||||
|
value was not set by the user */
|
||||||
|
|
||||||
update_auto_increment();
|
update_auto_increment();
|
||||||
|
}
|
||||||
if (auto_inc == 0) {
|
|
||||||
/* The autoinc counter for our table was not yet
|
|
||||||
initialized, initialize it now */
|
|
||||||
|
|
||||||
auto_inc = table->next_number_field->val_int();
|
|
||||||
|
|
||||||
srv_conc_enter_innodb(prebuilt->trx);
|
|
||||||
error = row_lock_table_autoinc_for_mysql(prebuilt);
|
|
||||||
srv_conc_exit_innodb(prebuilt->trx);
|
|
||||||
|
|
||||||
if (error != DB_SUCCESS) {
|
|
||||||
|
|
||||||
error = convert_error_code_to_mysql(error,
|
|
||||||
user_thd);
|
|
||||||
goto func_exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
dict_table_autoinc_initialize(prebuilt->table,
|
|
||||||
auto_inc);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* We have to set sql_stat_start to TRUE because
|
|
||||||
update_auto_increment may have called a select, and
|
|
||||||
has reset that flag; row_insert_for_mysql has to
|
|
||||||
know to set the IX intention lock on the table, something
|
|
||||||
it only does at the start of each statement */
|
|
||||||
|
|
||||||
prebuilt->sql_stat_start = TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (prebuilt->mysql_template == NULL
|
if (prebuilt->mysql_template == NULL
|
||||||
|| prebuilt->template_type != ROW_MYSQL_WHOLE_ROW) {
|
|| prebuilt->template_type != ROW_MYSQL_WHOLE_ROW) {
|
||||||
|
@ -1901,6 +1919,55 @@ convert_search_mode_to_innobase(
|
||||||
return(0);
|
return(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
BACKGROUND INFO: HOW A SELECT SQL QUERY IS EXECUTED
|
||||||
|
---------------------------------------------------
|
||||||
|
The following does not cover all the details, but explains how we determine
|
||||||
|
the start of a new SQL statement, and what is associated with it.
|
||||||
|
|
||||||
|
For each table in the database the MySQL interpreter may have several
|
||||||
|
table handle instances in use, also in a single SQL query. For each table
|
||||||
|
handle instance there is an InnoDB 'prebuilt' struct which contains most
|
||||||
|
of the InnoDB data associated with this table handle instance.
|
||||||
|
|
||||||
|
A) if the user has not explicitly set any MySQL table level locks:
|
||||||
|
|
||||||
|
1) MySQL calls ::external_lock to set an 'intention' table level lock on
|
||||||
|
the table of the handle instance. There we set
|
||||||
|
prebuilt->sql_stat_start = TRUE. The flag sql_stat_start should be set
|
||||||
|
true if we are taking this table handle instance to use in a new SQL
|
||||||
|
statement issued by the user. We also increment trx->n_mysql_tables_in_use.
|
||||||
|
|
||||||
|
2) If prebuilt->sql_stat_start == TRUE we 'pre-compile' the MySQL search
|
||||||
|
instructions to prebuilt->template of the table handle instance in
|
||||||
|
::index_read. The template is used to save CPU time in large joins.
|
||||||
|
|
||||||
|
3) In row_search_for_mysql, if prebuilt->sql_stat_start is true, we
|
||||||
|
allocate a new consistent read view for the trx if it does not yet have one,
|
||||||
|
or in the case of a locking read, set an InnoDB 'intention' table level
|
||||||
|
lock on the table.
|
||||||
|
|
||||||
|
4) We do the SELECT. MySQL may repeatedly call ::index_read for the
|
||||||
|
same table handle instance, if it is a join.
|
||||||
|
|
||||||
|
5) When the SELECT ends, MySQL removes its intention table level locks
|
||||||
|
in ::external_lock. When trx->n_mysql_tables_in_use drops to zero,
|
||||||
|
(a) we execute a COMMIT there if the autocommit is on,
|
||||||
|
(b) we also release possible 'SQL statement level resources' InnoDB may
|
||||||
|
have for this SQL statement. The MySQL interpreter does NOT execute
|
||||||
|
autocommit for pure read transactions, though it should. That is why the
|
||||||
|
table handler in that case has to execute the COMMIT in ::external_lock.
|
||||||
|
|
||||||
|
B) If the user has explicitly set MySQL table level locks, then MySQL
|
||||||
|
does NOT call ::external_lock at the start of the statement. To determine
|
||||||
|
when we are at the start of a new SQL statement we at the start of
|
||||||
|
::index_read also compare the query id to the latest query id where the
|
||||||
|
table handle instance was used. If it has changed, we know we are at the
|
||||||
|
start of a new SQL statement. Since the query id can theoretically
|
||||||
|
overwrap, we use this test only as a secondary way of determining the
|
||||||
|
start of a new SQL statement. */
|
||||||
|
|
||||||
|
|
||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
Positions an index cursor to the index specified in the handle. Fetches the
|
Positions an index cursor to the index specified in the handle. Fetches the
|
||||||
row if any. */
|
row if any. */
|
||||||
|
@ -1914,7 +1981,10 @@ ha_innobase::index_read(
|
||||||
row */
|
row */
|
||||||
const mysql_byte* key_ptr,/* in: key value; if this is NULL
|
const mysql_byte* key_ptr,/* in: key value; if this is NULL
|
||||||
we position the cursor at the
|
we position the cursor at the
|
||||||
start or end of index */
|
start or end of index; this can
|
||||||
|
also contain an InnoDB row id, in
|
||||||
|
which case key_len is the InnoDB
|
||||||
|
row id length */
|
||||||
uint key_len,/* in: key value length */
|
uint key_len,/* in: key value length */
|
||||||
enum ha_rkey_function find_flag)/* in: search flags from my_base.h */
|
enum ha_rkey_function find_flag)/* in: search flags from my_base.h */
|
||||||
{
|
{
|
||||||
|
@ -1941,10 +2011,8 @@ ha_innobase::index_read(
|
||||||
|
|
||||||
index = prebuilt->index;
|
index = prebuilt->index;
|
||||||
|
|
||||||
/* Note that if the select is used for an update, we always
|
/* Note that if the index for which the search template is built is not
|
||||||
fetch the clustered index record: therefore the index for which the
|
necessarily prebuilt->index, but can also be the clustered index */
|
||||||
template is built is not necessarily prebuilt->index, but can also
|
|
||||||
be the clustered index */
|
|
||||||
|
|
||||||
if (prebuilt->sql_stat_start) {
|
if (prebuilt->sql_stat_start) {
|
||||||
build_template(prebuilt, user_thd, table,
|
build_template(prebuilt, user_thd, table,
|
||||||
|
@ -1952,6 +2020,9 @@ ha_innobase::index_read(
|
||||||
}
|
}
|
||||||
|
|
||||||
if (key_ptr) {
|
if (key_ptr) {
|
||||||
|
/* Convert the search key value to InnoDB format into
|
||||||
|
prebuilt->search_tuple */
|
||||||
|
|
||||||
row_sel_convert_mysql_key_to_innobase(prebuilt->search_tuple,
|
row_sel_convert_mysql_key_to_innobase(prebuilt->search_tuple,
|
||||||
(byte*) key_val_buff,
|
(byte*) key_val_buff,
|
||||||
index,
|
index,
|
||||||
|
@ -2313,8 +2384,7 @@ ha_innobase::rnd_next(
|
||||||
}
|
}
|
||||||
|
|
||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
Fetches a row from the table based on a reference. TODO: currently we use
|
Fetches a row from the table based on a row reference. */
|
||||||
'ref_stored_len' of the handle as the key length. This may change. */
|
|
||||||
|
|
||||||
int
|
int
|
||||||
ha_innobase::rnd_pos(
|
ha_innobase::rnd_pos(
|
||||||
|
@ -2322,13 +2392,17 @@ ha_innobase::rnd_pos(
|
||||||
/* out: 0, HA_ERR_KEY_NOT_FOUND,
|
/* out: 0, HA_ERR_KEY_NOT_FOUND,
|
||||||
or error code */
|
or error code */
|
||||||
mysql_byte* buf, /* in/out: buffer for the row */
|
mysql_byte* buf, /* in/out: buffer for the row */
|
||||||
mysql_byte* pos) /* in: primary key value in MySQL format */
|
mysql_byte* pos) /* in: primary key value of the row in the
|
||||||
|
MySQL format, or the row id if the clustered
|
||||||
|
index was internally generated by InnoDB;
|
||||||
|
the length of data in pos has to be
|
||||||
|
ref_length */
|
||||||
{
|
{
|
||||||
row_prebuilt_t* prebuilt = (row_prebuilt_t*) innobase_prebuilt;
|
row_prebuilt_t* prebuilt = (row_prebuilt_t*) innobase_prebuilt;
|
||||||
int error;
|
int error;
|
||||||
uint keynr = active_index;
|
uint keynr = active_index;
|
||||||
DBUG_ENTER("rnd_pos");
|
DBUG_ENTER("rnd_pos");
|
||||||
DBUG_DUMP("key", (char*) pos, ref_stored_len);
|
DBUG_DUMP("key", (char*) pos, ref_length);
|
||||||
|
|
||||||
statistic_increment(ha_read_rnd_count, &LOCK_status);
|
statistic_increment(ha_read_rnd_count, &LOCK_status);
|
||||||
|
|
||||||
|
@ -2339,7 +2413,7 @@ ha_innobase::rnd_pos(
|
||||||
/* No primary key was defined for the table and we
|
/* No primary key was defined for the table and we
|
||||||
generated the clustered index from the row id: the
|
generated the clustered index from the row id: the
|
||||||
row reference is the row id, not any key value
|
row reference is the row id, not any key value
|
||||||
that MySQL knows */
|
that MySQL knows of */
|
||||||
|
|
||||||
error = change_active_index(MAX_KEY);
|
error = change_active_index(MAX_KEY);
|
||||||
} else {
|
} else {
|
||||||
|
@ -2351,7 +2425,10 @@ ha_innobase::rnd_pos(
|
||||||
DBUG_RETURN(error);
|
DBUG_RETURN(error);
|
||||||
}
|
}
|
||||||
|
|
||||||
error = index_read(buf, pos, ref_stored_len, HA_READ_KEY_EXACT);
|
/* Note that we assume the length of the row reference is fixed
|
||||||
|
for the table, and it is == ref_length */
|
||||||
|
|
||||||
|
error = index_read(buf, pos, ref_length, HA_READ_KEY_EXACT);
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
DBUG_PRINT("error",("Got error: %ld",error));
|
DBUG_PRINT("error",("Got error: %ld",error));
|
||||||
|
@ -2363,8 +2440,8 @@ ha_innobase::rnd_pos(
|
||||||
|
|
||||||
/*************************************************************************
|
/*************************************************************************
|
||||||
Stores a reference to the current row to 'ref' field of the handle. Note
|
Stores a reference to the current row to 'ref' field of the handle. Note
|
||||||
that the function parameter is illogical: we must assume that 'record'
|
that in the case where we have generated the clustered index for the
|
||||||
is the current 'position' of the handle, because if row ref is actually
|
table, the function parameter is illogical: we MUST ASSUME that 'record'
|
||||||
the row id internally generated in InnoDB, then 'record' does not contain
|
the row id internally generated in InnoDB, then 'record' does not contain
|
||||||
it. We just guess that the row id must be for the record where the handle
|
it. We just guess that the row id must be for the record where the handle
|
||||||
was positioned the last time. */
|
was positioned the last time. */
|
||||||
|
@ -2384,7 +2461,7 @@ ha_innobase::position(
|
||||||
/* No primary key was defined for the table and we
|
/* No primary key was defined for the table and we
|
||||||
generated the clustered index from row id: the
|
generated the clustered index from row id: the
|
||||||
row reference will be the row id, not any key value
|
row reference will be the row id, not any key value
|
||||||
that MySQL knows */
|
that MySQL knows of */
|
||||||
|
|
||||||
len = DATA_ROW_ID_LEN;
|
len = DATA_ROW_ID_LEN;
|
||||||
|
|
||||||
|
@ -2393,8 +2470,11 @@ ha_innobase::position(
|
||||||
len = store_key_val_for_row(primary_key, (char*) ref, record);
|
len = store_key_val_for_row(primary_key, (char*) ref, record);
|
||||||
}
|
}
|
||||||
|
|
||||||
DBUG_ASSERT(len == ref_length);
|
/* Since we do not store len to the buffer 'ref', we must assume
|
||||||
ref_stored_len = len;
|
that len is always fixed for this table. The following assertion
|
||||||
|
checks this. */
|
||||||
|
|
||||||
|
ut_a(len == ref_length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -2612,7 +2692,7 @@ ha_innobase::create(
|
||||||
/* Our function row_get_mysql_key_number_for_index assumes
|
/* Our function row_get_mysql_key_number_for_index assumes
|
||||||
the primary key is always number 0, if it exists */
|
the primary key is always number 0, if it exists */
|
||||||
|
|
||||||
assert(primary_key_no == -1 || primary_key_no == 0);
|
DBUG_ASSERT(primary_key_no == -1 || primary_key_no == 0);
|
||||||
|
|
||||||
/* Create the keys */
|
/* Create the keys */
|
||||||
|
|
||||||
|
@ -2693,7 +2773,7 @@ ha_innobase::create(
|
||||||
|
|
||||||
innobase_table = dict_table_get(norm_name, NULL);
|
innobase_table = dict_table_get(norm_name, NULL);
|
||||||
|
|
||||||
assert(innobase_table != 0);
|
DBUG_ASSERT(innobase_table != 0);
|
||||||
|
|
||||||
/* Tell the InnoDB server that there might be work for
|
/* Tell the InnoDB server that there might be work for
|
||||||
utility threads: */
|
utility threads: */
|
||||||
|
@ -3519,37 +3599,53 @@ ha_innobase::store_lock(
|
||||||
}
|
}
|
||||||
|
|
||||||
/***********************************************************************
|
/***********************************************************************
|
||||||
Returns the next auto-increment column value for the table. write_row
|
This function initializes the auto-inc counter if it has not been
|
||||||
normally fetches the value from the cache in the data dictionary. This
|
initialized yet. This function does not change the value of the auto-inc
|
||||||
function in used by SHOW TABLE STATUS and when the first insert to the table
|
counter if it already has been initialized. In parameter ret returns
|
||||||
is done after database startup. */
|
the value of the auto-inc counter. */
|
||||||
|
|
||||||
longlong
|
int
|
||||||
ha_innobase::get_auto_increment()
|
ha_innobase::innobase_read_and_init_auto_inc(
|
||||||
/*=============================*/
|
/*=========================================*/
|
||||||
/* out: the next auto-increment column value */
|
/* out: 0 or error code: deadlock or
|
||||||
|
lock wait timeout */
|
||||||
|
longlong* ret) /* out: auto-inc value */
|
||||||
{
|
{
|
||||||
row_prebuilt_t* prebuilt = (row_prebuilt_t*) innobase_prebuilt;
|
row_prebuilt_t* prebuilt = (row_prebuilt_t*) innobase_prebuilt;
|
||||||
longlong nr;
|
longlong auto_inc;
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
|
ut_a(prebuilt);
|
||||||
ut_a(prebuilt->trx ==
|
ut_a(prebuilt->trx ==
|
||||||
(trx_t*) current_thd->transaction.all.innobase_tid);
|
(trx_t*) current_thd->transaction.all.innobase_tid);
|
||||||
|
ut_a(prebuilt->table);
|
||||||
|
|
||||||
|
auto_inc = dict_table_autoinc_read(prebuilt->table);
|
||||||
|
|
||||||
/* Also SHOW TABLE STATUS calls this function. Previously, when we did
|
if (auto_inc != 0) {
|
||||||
always read the max autoinc key value, setting x-locks, users were
|
/* Already initialized */
|
||||||
surprised that SHOW TABLE STATUS could end up in a deadlock with
|
*ret = auto_inc;
|
||||||
ordinary SQL queries. We avoid these deadlocks if the auto-inc
|
|
||||||
counter for the table has been initialized by fetching the value
|
return(0);
|
||||||
from the table struct in dictionary cache. */
|
}
|
||||||
|
|
||||||
assert(prebuilt->table);
|
srv_conc_enter_innodb(prebuilt->trx);
|
||||||
|
error = row_lock_table_autoinc_for_mysql(prebuilt);
|
||||||
nr = dict_table_autoinc_read(prebuilt->table);
|
srv_conc_exit_innodb(prebuilt->trx);
|
||||||
|
|
||||||
if (nr != 0) {
|
if (error != DB_SUCCESS) {
|
||||||
|
error = convert_error_code_to_mysql(error, user_thd);
|
||||||
|
|
||||||
return(nr + 1);
|
goto func_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check again if someone has initialized the counter meanwhile */
|
||||||
|
auto_inc = dict_table_autoinc_read(prebuilt->table);
|
||||||
|
|
||||||
|
if (auto_inc != 0) {
|
||||||
|
*ret = auto_inc;
|
||||||
|
|
||||||
|
return(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
(void) extra(HA_EXTRA_KEYREAD);
|
(void) extra(HA_EXTRA_KEYREAD);
|
||||||
|
@ -3569,22 +3665,63 @@ ha_innobase::get_auto_increment()
|
||||||
|
|
||||||
prebuilt->hint_no_need_to_fetch_extra_cols = FALSE;
|
prebuilt->hint_no_need_to_fetch_extra_cols = FALSE;
|
||||||
|
|
||||||
prebuilt->trx->mysql_n_tables_locked += 1;
|
prebuilt->trx->mysql_n_tables_locked += 1;
|
||||||
|
|
||||||
error = index_last(table->record[1]);
|
error = index_last(table->record[1]);
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
nr = 1;
|
if (error == HA_ERR_END_OF_FILE) {
|
||||||
|
/* The table was empty, initialize to 1 */
|
||||||
|
auto_inc = 1;
|
||||||
|
|
||||||
|
error = 0;
|
||||||
|
} else {
|
||||||
|
/* Deadlock or a lock wait timeout */
|
||||||
|
auto_inc = -1;
|
||||||
|
|
||||||
|
goto func_exit;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
nr = (longlong) table->next_number_field->
|
/* Initialize to max(col) + 1 */
|
||||||
|
auto_inc = (longlong) table->next_number_field->
|
||||||
val_int_offset(table->rec_buff_length) + 1;
|
val_int_offset(table->rec_buff_length) + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dict_table_autoinc_initialize(prebuilt->table, auto_inc);
|
||||||
|
|
||||||
|
func_exit:
|
||||||
(void) extra(HA_EXTRA_NO_KEYREAD);
|
(void) extra(HA_EXTRA_NO_KEYREAD);
|
||||||
|
|
||||||
index_end();
|
index_end();
|
||||||
|
|
||||||
return(nr);
|
*ret = auto_inc;
|
||||||
|
|
||||||
|
return(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
/***********************************************************************
|
||||||
|
This function initializes the auto-inc counter if it has not been
|
||||||
|
initialized yet. This function does not change the value of the auto-inc
|
||||||
|
counter if it already has been initialized. Returns the value of the
|
||||||
|
auto-inc counter. */
|
||||||
|
|
||||||
|
longlong
|
||||||
|
ha_innobase::get_auto_increment()
|
||||||
|
/*=============================*/
|
||||||
|
/* out: auto-increment column value, -1 if error
|
||||||
|
(deadlock or lock wait timeout) */
|
||||||
|
{
|
||||||
|
longlong nr;
|
||||||
|
int error;
|
||||||
|
|
||||||
|
error = innobase_read_and_init_auto_inc(&nr);
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return(nr);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* HAVE_INNOBASE_DB */
|
#endif /* HAVE_INNOBASE_DB */
|
||||||
|
|
|
@ -52,8 +52,6 @@ class ha_innobase: public handler
|
||||||
byte* key_val_buff; /* buffer used in converting
|
byte* key_val_buff; /* buffer used in converting
|
||||||
search key values from MySQL format
|
search key values from MySQL format
|
||||||
to Innodb format */
|
to Innodb format */
|
||||||
uint ref_stored_len; /* length of the key value stored to
|
|
||||||
'ref' buffer of the handle, if any */
|
|
||||||
ulong int_table_flags;
|
ulong int_table_flags;
|
||||||
uint primary_key;
|
uint primary_key;
|
||||||
uint last_dup_key;
|
uint last_dup_key;
|
||||||
|
@ -71,6 +69,7 @@ class ha_innobase: public handler
|
||||||
int update_thd(THD* thd);
|
int update_thd(THD* thd);
|
||||||
int change_active_index(uint keynr);
|
int change_active_index(uint keynr);
|
||||||
int general_fetch(byte* buf, uint direction, uint match_mode);
|
int general_fetch(byte* buf, uint direction, uint match_mode);
|
||||||
|
int innobase_read_and_init_auto_inc(longlong* ret);
|
||||||
|
|
||||||
/* Init values for the class: */
|
/* Init values for the class: */
|
||||||
public:
|
public:
|
||||||
|
|
|
@ -251,7 +251,6 @@ static SYMBOL symbols[] = {
|
||||||
{ "NEW", SYM(NEW_SYM),0,0},
|
{ "NEW", SYM(NEW_SYM),0,0},
|
||||||
{ "NCHAR", SYM(NCHAR_SYM),0,0},
|
{ "NCHAR", SYM(NCHAR_SYM),0,0},
|
||||||
{ "NO", SYM(NO_SYM),0,0},
|
{ "NO", SYM(NO_SYM),0,0},
|
||||||
{ "NO_FOREIGN_KEY_CHECKS", SYM(NO_FOREIGN_KEY_CHECKS), 0, 0},
|
|
||||||
{ "NOT", SYM(NOT),0,0},
|
{ "NOT", SYM(NOT),0,0},
|
||||||
{ "NULL", SYM(NULL_SYM),0,0},
|
{ "NULL", SYM(NULL_SYM),0,0},
|
||||||
{ "NUMERIC", SYM(NUMERIC_SYM),0,0},
|
{ "NUMERIC", SYM(NUMERIC_SYM),0,0},
|
||||||
|
@ -285,7 +284,6 @@ static SYMBOL symbols[] = {
|
||||||
{ "RELAY_LOG_POS", SYM(RELAY_LOG_POS_SYM),0,0},
|
{ "RELAY_LOG_POS", SYM(RELAY_LOG_POS_SYM),0,0},
|
||||||
{ "RELOAD", SYM(RELOAD),0,0},
|
{ "RELOAD", SYM(RELOAD),0,0},
|
||||||
{ "REGEXP", SYM(REGEXP),0,0},
|
{ "REGEXP", SYM(REGEXP),0,0},
|
||||||
{ "RELAXED_UNIQUE_CHECKS", SYM(RELAXED_UNIQUE_CHECKS), 0, 0},
|
|
||||||
{ "RENAME", SYM(RENAME),0,0},
|
{ "RENAME", SYM(RENAME),0,0},
|
||||||
{ "REPAIR", SYM(REPAIR),0,0},
|
{ "REPAIR", SYM(REPAIR),0,0},
|
||||||
{ "REPLACE", SYM(REPLACE),0,0},
|
{ "REPLACE", SYM(REPLACE),0,0},
|
||||||
|
|
10
sql/log.cc
10
sql/log.cc
|
@ -1320,16 +1320,16 @@ bool MYSQL_LOG::write(THD *thd,const char *query, uint query_length,
|
||||||
if (end != buff)
|
if (end != buff)
|
||||||
{
|
{
|
||||||
*end++=';';
|
*end++=';';
|
||||||
*end++='\n';
|
*end='\n';
|
||||||
*end=0;
|
|
||||||
if (my_b_write(&log_file, (byte*) "SET ",4) ||
|
if (my_b_write(&log_file, (byte*) "SET ",4) ||
|
||||||
my_b_write(&log_file, (byte*) buff+1,(uint) (end-buff)-1))
|
my_b_write(&log_file, (byte*) buff+1,(uint) (end-buff)))
|
||||||
tmp_errno=errno;
|
tmp_errno=errno;
|
||||||
}
|
}
|
||||||
if (!query)
|
if (!query)
|
||||||
{
|
{
|
||||||
query="#adminstrator command";
|
end=strxmov(buff, "# administrator command: ",
|
||||||
query_length=21;
|
command_name[thd->command], NullS);
|
||||||
|
query_length=(ulong) (end-buff);
|
||||||
}
|
}
|
||||||
if (my_b_write(&log_file, (byte*) query,query_length) ||
|
if (my_b_write(&log_file, (byte*) query,query_length) ||
|
||||||
my_b_write(&log_file, (byte*) ";\n",2) ||
|
my_b_write(&log_file, (byte*) ";\n",2) ||
|
||||||
|
|
|
@ -1809,7 +1809,7 @@ int main(int argc, char **argv)
|
||||||
exit( 1 );
|
exit( 1 );
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
load_defaults("my",load_default_groups,&argc,&argv);
|
load_defaults(MYSQL_CONFIG_NAME,load_default_groups,&argc,&argv);
|
||||||
defaults_argv=argv;
|
defaults_argv=argv;
|
||||||
|
|
||||||
/* Get default temporary directory */
|
/* Get default temporary directory */
|
||||||
|
@ -2373,7 +2373,15 @@ static void create_new_thread(THD *thd)
|
||||||
for (uint i=0; i < 8 ; i++) // Generate password teststring
|
for (uint i=0; i < 8 ; i++) // Generate password teststring
|
||||||
thd->scramble[i]= (char) (rnd(&sql_rand)*94+33);
|
thd->scramble[i]= (char) (rnd(&sql_rand)*94+33);
|
||||||
thd->scramble[8]=0;
|
thd->scramble[8]=0;
|
||||||
thd->rand=sql_rand;
|
/*
|
||||||
|
We need good random number initialization for new thread
|
||||||
|
Just coping global one will not work
|
||||||
|
*/
|
||||||
|
{
|
||||||
|
ulong tmp=(ulong) (rnd(&sql_rand) * 3000000);
|
||||||
|
randominit(&(thd->rand), tmp + (ulong) start_time,
|
||||||
|
tmp + (ulong) thread_id);
|
||||||
|
}
|
||||||
thd->real_id=pthread_self(); // Keep purify happy
|
thd->real_id=pthread_self(); // Keep purify happy
|
||||||
|
|
||||||
/* Start a new thread to handle connection */
|
/* Start a new thread to handle connection */
|
||||||
|
@ -3260,8 +3268,8 @@ struct my_option my_long_options[] =
|
||||||
#endif
|
#endif
|
||||||
{"temp-pool", OPT_TEMP_POOL,
|
{"temp-pool", OPT_TEMP_POOL,
|
||||||
"Using this option will cause most temporary files created to use a small set of names, rather than a unique name for each new file.",
|
"Using this option will cause most temporary files created to use a small set of names, rather than a unique name for each new file.",
|
||||||
(gptr*) &use_temp_pool, (gptr*) &use_temp_pool, 0, GET_BOOL, NO_ARG, 0, 0,
|
(gptr*) &use_temp_pool, (gptr*) &use_temp_pool, 0, GET_BOOL, NO_ARG, 1,
|
||||||
0, 0, 0, 0},
|
0, 0, 0, 0, 0},
|
||||||
{"tmpdir", 't', "Path for temporary files", (gptr*) &opt_mysql_tmpdir,
|
{"tmpdir", 't', "Path for temporary files", (gptr*) &opt_mysql_tmpdir,
|
||||||
(gptr*) &opt_mysql_tmpdir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
|
(gptr*) &opt_mysql_tmpdir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
|
||||||
{"transaction-isolation", OPT_TX_ISOLATION,
|
{"transaction-isolation", OPT_TX_ISOLATION,
|
||||||
|
@ -3805,7 +3813,7 @@ Starts the MySQL server\n");
|
||||||
");
|
");
|
||||||
puts("");
|
puts("");
|
||||||
#endif
|
#endif
|
||||||
print_defaults("my",load_default_groups);
|
print_defaults(MYSQL_CONFIG_NAME,load_default_groups);
|
||||||
puts("");
|
puts("");
|
||||||
fix_paths();
|
fix_paths();
|
||||||
set_ports();
|
set_ports();
|
||||||
|
|
|
@ -23,6 +23,7 @@
|
||||||
#include "sql_repl.h"
|
#include "sql_repl.h"
|
||||||
#include "repl_failsafe.h"
|
#include "repl_failsafe.h"
|
||||||
#include <thr_alarm.h>
|
#include <thr_alarm.h>
|
||||||
|
#include <my_dir.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
bool use_slave_mask = 0;
|
bool use_slave_mask = 0;
|
||||||
|
@ -1029,7 +1030,6 @@ void end_master_info(MASTER_INFO* mi)
|
||||||
|
|
||||||
int init_relay_log_info(RELAY_LOG_INFO* rli, const char* info_fname)
|
int init_relay_log_info(RELAY_LOG_INFO* rli, const char* info_fname)
|
||||||
{
|
{
|
||||||
MY_STAT stat_area;
|
|
||||||
char fname[FN_REFLEN+128];
|
char fname[FN_REFLEN+128];
|
||||||
int info_fd;
|
int info_fd;
|
||||||
const char* msg = 0;
|
const char* msg = 0;
|
||||||
|
@ -1069,7 +1069,7 @@ int init_relay_log_info(RELAY_LOG_INFO* rli, const char* info_fname)
|
||||||
DBUG_RETURN(1);
|
DBUG_RETURN(1);
|
||||||
|
|
||||||
/* if file does not exist */
|
/* if file does not exist */
|
||||||
if (!my_stat(fname, &stat_area, MYF(0)))
|
if (access(fname,F_OK))
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
If someone removed the file from underneath our feet, just close
|
If someone removed the file from underneath our feet, just close
|
||||||
|
|
|
@ -253,7 +253,7 @@ int acl_init(bool dont_read_acl_tables)
|
||||||
continue; /* purecov: tested */
|
continue; /* purecov: tested */
|
||||||
}
|
}
|
||||||
get_salt_from_password(user.salt,user.password);
|
get_salt_from_password(user.salt,user.password);
|
||||||
user.access=get_access(table,3);
|
user.access=get_access(table,3) & GLOBAL_ACLS;
|
||||||
user.sort=get_sort(2,user.host.hostname,user.user);
|
user.sort=get_sort(2,user.host.hostname,user.user);
|
||||||
user.hostname_length= (user.host.hostname ?
|
user.hostname_length= (user.host.hostname ?
|
||||||
(uint) strlen(user.host.hostname) : 0);
|
(uint) strlen(user.host.hostname) : 0);
|
||||||
|
@ -321,6 +321,11 @@ int acl_init(bool dont_read_acl_tables)
|
||||||
ACL_DB db;
|
ACL_DB db;
|
||||||
update_hostname(&db.host,get_field(&mem, table,0));
|
update_hostname(&db.host,get_field(&mem, table,0));
|
||||||
db.db=get_field(&mem, table,1);
|
db.db=get_field(&mem, table,1);
|
||||||
|
if (!db.db)
|
||||||
|
{
|
||||||
|
sql_print_error("Found an entry in the 'db' table with empty database name; Skipped");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
db.user=get_field(&mem, table,2);
|
db.user=get_field(&mem, table,2);
|
||||||
db.access=get_access(table,3);
|
db.access=get_access(table,3);
|
||||||
db.access=fix_rights_for_db(db.access);
|
db.access=fix_rights_for_db(db.access);
|
||||||
|
|
|
@ -32,7 +32,7 @@ TABLE *unused_tables; /* Used by mysql_test */
|
||||||
HASH open_cache; /* Used by mysql_test */
|
HASH open_cache; /* Used by mysql_test */
|
||||||
|
|
||||||
static int open_unireg_entry(THD *thd,TABLE *entry,const char *db,
|
static int open_unireg_entry(THD *thd,TABLE *entry,const char *db,
|
||||||
const char *name, const char *alias, bool locked);
|
const char *name, const char *alias);
|
||||||
static void free_cache_entry(TABLE *entry);
|
static void free_cache_entry(TABLE *entry);
|
||||||
static void mysql_rm_tmp_tables(void);
|
static void mysql_rm_tmp_tables(void);
|
||||||
static key_map get_key_map_from_key_list(TABLE *table,
|
static key_map get_key_map_from_key_list(TABLE *table,
|
||||||
|
@ -108,6 +108,24 @@ static void check_unused(void)
|
||||||
#define check_unused()
|
#define check_unused()
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
Create a list for all open tables matching SQL expression
|
||||||
|
|
||||||
|
SYNOPSIS
|
||||||
|
list_open_tables()
|
||||||
|
thd Thread THD
|
||||||
|
wild SQL like expression
|
||||||
|
|
||||||
|
NOTES
|
||||||
|
One gets only a list of tables for which one has any kind of privilege.
|
||||||
|
db and table names are allocated in result struct, so one doesn't need
|
||||||
|
a lock on LOCK_open when traversing the return list.
|
||||||
|
|
||||||
|
RETURN VALUES
|
||||||
|
NULL Error (Probably OOM)
|
||||||
|
# Pointer to list of names of open tables.
|
||||||
|
*/
|
||||||
|
|
||||||
OPEN_TABLE_LIST *list_open_tables(THD *thd, const char *wild)
|
OPEN_TABLE_LIST *list_open_tables(THD *thd, const char *wild)
|
||||||
{
|
{
|
||||||
int result = 0;
|
int result = 0;
|
||||||
|
@ -275,6 +293,16 @@ void intern_close_table(TABLE *table)
|
||||||
VOID(closefrm(table)); // close file
|
VOID(closefrm(table)); // close file
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Remove table from the open table cache
|
||||||
|
|
||||||
|
SYNOPSIS
|
||||||
|
free_cache_entry()
|
||||||
|
table Table to remove
|
||||||
|
|
||||||
|
NOTE
|
||||||
|
We need to have a lock on LOCK_open when calling this
|
||||||
|
*/
|
||||||
|
|
||||||
static void free_cache_entry(TABLE *table)
|
static void free_cache_entry(TABLE *table)
|
||||||
{
|
{
|
||||||
|
@ -709,7 +737,7 @@ TABLE *reopen_name_locked_table(THD* thd, TABLE_LIST* table_list)
|
||||||
key_length=(uint) (strmov(strmov(key,db)+1,table_name)-key)+1;
|
key_length=(uint) (strmov(strmov(key,db)+1,table_name)-key)+1;
|
||||||
|
|
||||||
pthread_mutex_lock(&LOCK_open);
|
pthread_mutex_lock(&LOCK_open);
|
||||||
if (open_unireg_entry(thd, table, db, table_name, table_name, 1) ||
|
if (open_unireg_entry(thd, table, db, table_name, table_name) ||
|
||||||
!(table->table_cache_key =memdup_root(&table->mem_root,(char*) key,
|
!(table->table_cache_key =memdup_root(&table->mem_root,(char*) key,
|
||||||
key_length)))
|
key_length)))
|
||||||
{
|
{
|
||||||
|
@ -842,8 +870,11 @@ TABLE *open_table(THD *thd,const char *db,const char *table_name,
|
||||||
|
|
||||||
/* make a new table */
|
/* make a new table */
|
||||||
if (!(table=(TABLE*) my_malloc(sizeof(*table),MYF(MY_WME))))
|
if (!(table=(TABLE*) my_malloc(sizeof(*table),MYF(MY_WME))))
|
||||||
|
{
|
||||||
|
VOID(pthread_mutex_unlock(&LOCK_open));
|
||||||
DBUG_RETURN(NULL);
|
DBUG_RETURN(NULL);
|
||||||
if (open_unireg_entry(thd, table,db,table_name,alias,1) ||
|
}
|
||||||
|
if (open_unireg_entry(thd, table,db,table_name,alias) ||
|
||||||
!(table->table_cache_key=memdup_root(&table->mem_root,(char*) key,
|
!(table->table_cache_key=memdup_root(&table->mem_root,(char*) key,
|
||||||
key_length)))
|
key_length)))
|
||||||
{
|
{
|
||||||
|
@ -934,8 +965,7 @@ bool reopen_table(TABLE *table,bool locked)
|
||||||
VOID(pthread_mutex_lock(&LOCK_open));
|
VOID(pthread_mutex_lock(&LOCK_open));
|
||||||
safe_mutex_assert_owner(&LOCK_open);
|
safe_mutex_assert_owner(&LOCK_open);
|
||||||
|
|
||||||
if (open_unireg_entry(current_thd,&tmp,db,table_name,table->table_name,
|
if (open_unireg_entry(current_thd,&tmp,db,table_name,table->table_name))
|
||||||
locked))
|
|
||||||
goto end;
|
goto end;
|
||||||
free_io_cache(table);
|
free_io_cache(table);
|
||||||
|
|
||||||
|
@ -1176,7 +1206,6 @@ bool wait_for_tables(THD *thd)
|
||||||
/* Now we can open all tables without any interference */
|
/* Now we can open all tables without any interference */
|
||||||
thd->proc_info="Reopen tables";
|
thd->proc_info="Reopen tables";
|
||||||
result=reopen_tables(thd,0,0);
|
result=reopen_tables(thd,0,0);
|
||||||
|
|
||||||
}
|
}
|
||||||
pthread_mutex_unlock(&LOCK_open);
|
pthread_mutex_unlock(&LOCK_open);
|
||||||
thd->proc_info=0;
|
thd->proc_info=0;
|
||||||
|
@ -1241,7 +1270,7 @@ void abort_locked_tables(THD *thd,const char *db, const char *table_name)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int open_unireg_entry(THD *thd, TABLE *entry, const char *db,
|
static int open_unireg_entry(THD *thd, TABLE *entry, const char *db,
|
||||||
const char *name, const char *alias, bool locked)
|
const char *name, const char *alias)
|
||||||
{
|
{
|
||||||
char path[FN_REFLEN];
|
char path[FN_REFLEN];
|
||||||
int error;
|
int error;
|
||||||
|
@ -1261,23 +1290,17 @@ static int open_unireg_entry(THD *thd, TABLE *entry, const char *db,
|
||||||
table_list.db=(char*) db;
|
table_list.db=(char*) db;
|
||||||
table_list.name=(char*) name;
|
table_list.name=(char*) name;
|
||||||
table_list.next=0;
|
table_list.next=0;
|
||||||
if (!locked)
|
|
||||||
pthread_mutex_lock(&LOCK_open);
|
|
||||||
safe_mutex_assert_owner(&LOCK_open);
|
safe_mutex_assert_owner(&LOCK_open);
|
||||||
|
|
||||||
if ((error=lock_table_name(thd,&table_list)))
|
if ((error=lock_table_name(thd,&table_list)))
|
||||||
{
|
{
|
||||||
if (error < 0)
|
if (error < 0)
|
||||||
{
|
{
|
||||||
if (!locked)
|
|
||||||
pthread_mutex_unlock(&LOCK_open);
|
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
if (wait_for_locked_table_names(thd,&table_list))
|
if (wait_for_locked_table_names(thd,&table_list))
|
||||||
{
|
{
|
||||||
unlock_table_name(thd,&table_list);
|
unlock_table_name(thd,&table_list);
|
||||||
if (!locked)
|
|
||||||
pthread_mutex_unlock(&LOCK_open);
|
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1307,9 +1330,9 @@ static int open_unireg_entry(THD *thd, TABLE *entry, const char *db,
|
||||||
thd->net.last_error[0]=0; // Clear error message
|
thd->net.last_error[0]=0; // Clear error message
|
||||||
thd->net.last_errno=0;
|
thd->net.last_errno=0;
|
||||||
}
|
}
|
||||||
if (locked)
|
pthread_mutex_lock(&LOCK_open);
|
||||||
pthread_mutex_lock(&LOCK_open); // Get back original lock
|
|
||||||
unlock_table_name(thd,&table_list);
|
unlock_table_name(thd,&table_list);
|
||||||
|
|
||||||
if (error)
|
if (error)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
@ -1368,9 +1391,9 @@ int open_tables(THD *thd,TABLE_LIST *start)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*prev_table=0;
|
*prev_table=0;
|
||||||
|
pthread_mutex_unlock(&LOCK_open);
|
||||||
if (found)
|
if (found)
|
||||||
VOID(pthread_cond_broadcast(&COND_refresh)); // Signal to refresh
|
VOID(pthread_cond_broadcast(&COND_refresh)); // Signal to refresh
|
||||||
pthread_mutex_unlock(&LOCK_open);
|
|
||||||
goto restart;
|
goto restart;
|
||||||
}
|
}
|
||||||
result= -1; // Fatal error
|
result= -1; // Fatal error
|
||||||
|
@ -2207,6 +2230,7 @@ int setup_ftfuncs(THD *thd)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int init_ftfuncs(THD *thd, bool no_order)
|
int init_ftfuncs(THD *thd, bool no_order)
|
||||||
{
|
{
|
||||||
if (thd->lex.select->ftfunc_list.elements)
|
if (thd->lex.select->ftfunc_list.elements)
|
||||||
|
@ -2217,9 +2241,7 @@ int init_ftfuncs(THD *thd, bool no_order)
|
||||||
thd->proc_info="FULLTEXT initialization";
|
thd->proc_info="FULLTEXT initialization";
|
||||||
|
|
||||||
while ((ifm=li++))
|
while ((ifm=li++))
|
||||||
{
|
|
||||||
ifm->init_search(no_order);
|
ifm->init_search(no_order);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1454,6 +1454,11 @@ bool select_create::send_eof()
|
||||||
table->file->extra(HA_EXTRA_NO_IGNORE_DUP_KEY);
|
table->file->extra(HA_EXTRA_NO_IGNORE_DUP_KEY);
|
||||||
VOID(pthread_mutex_lock(&LOCK_open));
|
VOID(pthread_mutex_lock(&LOCK_open));
|
||||||
mysql_unlock_tables(thd, lock);
|
mysql_unlock_tables(thd, lock);
|
||||||
|
/*
|
||||||
|
TODO:
|
||||||
|
Check if we can remove the following two rows.
|
||||||
|
We should be able to just keep the table in the table cache.
|
||||||
|
*/
|
||||||
if (!table->tmp_table)
|
if (!table->tmp_table)
|
||||||
hash_delete(&open_cache,(byte*) table);
|
hash_delete(&open_cache,(byte*) table);
|
||||||
lock=0;
|
lock=0;
|
||||||
|
|
|
@ -1493,7 +1493,7 @@ mysql_execute_command(void)
|
||||||
/* Check that the first table has CREATE privilege */
|
/* Check that the first table has CREATE privilege */
|
||||||
TABLE_LIST *tmp_table_list=tables->next;
|
TABLE_LIST *tmp_table_list=tables->next;
|
||||||
tables->next=0;
|
tables->next=0;
|
||||||
bool error=check_grant(thd,CREATE_ACL,tables);
|
bool error=check_grant(thd, want_priv, tables);
|
||||||
tables->next=tmp_table_list;
|
tables->next=tmp_table_list;
|
||||||
if (error)
|
if (error)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
|
@ -3674,7 +3674,7 @@ create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List<Item> &fields,
|
||||||
field_count= (uint) (reg_field - table->field);
|
field_count= (uint) (reg_field - table->field);
|
||||||
|
|
||||||
/* If result table is small; use a heap */
|
/* If result table is small; use a heap */
|
||||||
if (blob_count || using_unique_constraint ||
|
if (blob_count || using_unique_constraint || group_null_items ||
|
||||||
(select_options & (OPTION_BIG_TABLES | SELECT_SMALL_RESULT)) ==
|
(select_options & (OPTION_BIG_TABLES | SELECT_SMALL_RESULT)) ==
|
||||||
OPTION_BIG_TABLES)
|
OPTION_BIG_TABLES)
|
||||||
{
|
{
|
||||||
|
|
|
@ -124,13 +124,15 @@ int mysqld_show_open_tables(THD *thd,const char *wild)
|
||||||
net_store_data(&thd->packet,open_list->in_use);
|
net_store_data(&thd->packet,open_list->in_use);
|
||||||
net_store_data(&thd->packet,open_list->locked);
|
net_store_data(&thd->packet,open_list->locked);
|
||||||
if (my_net_write(&thd->net,(char*) thd->packet.ptr(),thd->packet.length()))
|
if (my_net_write(&thd->net,(char*) thd->packet.ptr(),thd->packet.length()))
|
||||||
|
{
|
||||||
DBUG_RETURN(-1);
|
DBUG_RETURN(-1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
send_eof(&thd->net);
|
send_eof(&thd->net);
|
||||||
DBUG_RETURN(0);
|
DBUG_RETURN(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/***************************************************************************
|
/***************************************************************************
|
||||||
** List all tables in a database (fast version)
|
** List all tables in a database (fast version)
|
||||||
** A table is a .frm file in the current databasedir
|
** A table is a .frm file in the current databasedir
|
||||||
|
|
|
@ -782,7 +782,9 @@ TABLE *create_table_from_items(THD *thd, HA_CREATE_INFO *create_info,
|
||||||
table->reginfo.lock_type=TL_WRITE;
|
table->reginfo.lock_type=TL_WRITE;
|
||||||
if (!((*lock)=mysql_lock_tables(thd,&table,1)))
|
if (!((*lock)=mysql_lock_tables(thd,&table,1)))
|
||||||
{
|
{
|
||||||
|
VOID(pthread_mutex_lock(&LOCK_open));
|
||||||
hash_delete(&open_cache,(byte*) table);
|
hash_delete(&open_cache,(byte*) table);
|
||||||
|
VOID(pthread_mutex_unlock(&LOCK_open));
|
||||||
quick_rm_table(create_info->db_type,db,name);
|
quick_rm_table(create_info->db_type,db,name);
|
||||||
DBUG_RETURN(0);
|
DBUG_RETURN(0);
|
||||||
}
|
}
|
||||||
|
@ -977,19 +979,25 @@ static int prepare_for_repair(THD* thd, TABLE_LIST* table,
|
||||||
|
|
||||||
if (my_rename(from, tmp, MYF(MY_WME)))
|
if (my_rename(from, tmp, MYF(MY_WME)))
|
||||||
{
|
{
|
||||||
|
pthread_mutex_lock(&LOCK_open);
|
||||||
unlock_table_name(thd, table);
|
unlock_table_name(thd, table);
|
||||||
|
pthread_mutex_unlock(&LOCK_open);
|
||||||
DBUG_RETURN(send_check_errmsg(thd, table, "repair",
|
DBUG_RETURN(send_check_errmsg(thd, table, "repair",
|
||||||
"Failed renaming .MYD file"));
|
"Failed renaming .MYD file"));
|
||||||
}
|
}
|
||||||
if (mysql_truncate(thd, table, 1))
|
if (mysql_truncate(thd, table, 1))
|
||||||
{
|
{
|
||||||
|
pthread_mutex_lock(&LOCK_open);
|
||||||
unlock_table_name(thd, table);
|
unlock_table_name(thd, table);
|
||||||
|
pthread_mutex_unlock(&LOCK_open);
|
||||||
DBUG_RETURN(send_check_errmsg(thd, table, "repair",
|
DBUG_RETURN(send_check_errmsg(thd, table, "repair",
|
||||||
"Failed generating table from .frm file"));
|
"Failed generating table from .frm file"));
|
||||||
}
|
}
|
||||||
if (my_rename(tmp, from, MYF(MY_WME)))
|
if (my_rename(tmp, from, MYF(MY_WME)))
|
||||||
{
|
{
|
||||||
|
pthread_mutex_lock(&LOCK_open);
|
||||||
unlock_table_name(thd, table);
|
unlock_table_name(thd, table);
|
||||||
|
pthread_mutex_unlock(&LOCK_open);
|
||||||
DBUG_RETURN(send_check_errmsg(thd, table, "repair",
|
DBUG_RETURN(send_check_errmsg(thd, table, "repair",
|
||||||
"Failed restoring .MYD file"));
|
"Failed restoring .MYD file"));
|
||||||
}
|
}
|
||||||
|
@ -1000,7 +1008,11 @@ static int prepare_for_repair(THD* thd, TABLE_LIST* table,
|
||||||
to finish the repair in the handler later on.
|
to finish the repair in the handler later on.
|
||||||
*/
|
*/
|
||||||
if (!(table->table = reopen_name_locked_table(thd, table)))
|
if (!(table->table = reopen_name_locked_table(thd, table)))
|
||||||
unlock_table_name(thd, table);
|
{
|
||||||
|
pthread_mutex_lock(&LOCK_open);
|
||||||
|
unlock_table_name(thd, table);
|
||||||
|
pthread_mutex_unlock(&LOCK_open);
|
||||||
|
}
|
||||||
DBUG_RETURN(0);
|
DBUG_RETURN(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1855,8 +1867,8 @@ int mysql_alter_table(THD *thd,char *new_db, char *new_name,
|
||||||
error=1;
|
error=1;
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
VOID(pthread_cond_broadcast(&COND_refresh));
|
|
||||||
VOID(pthread_mutex_unlock(&LOCK_open));
|
VOID(pthread_mutex_unlock(&LOCK_open));
|
||||||
|
VOID(pthread_cond_broadcast(&COND_refresh));
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
#ifdef HAVE_BERKELEY_DB
|
#ifdef HAVE_BERKELEY_DB
|
||||||
|
|
|
@ -44,7 +44,6 @@ noinst_PROGRAMS = conf_to_src
|
||||||
EXTRA_DIST = ctype-big5.c ctype-czech.c ctype-euc_kr.c \
|
EXTRA_DIST = ctype-big5.c ctype-czech.c ctype-euc_kr.c \
|
||||||
ctype-gb2312.c ctype-gbk.c ctype-sjis.c \
|
ctype-gb2312.c ctype-gbk.c ctype-sjis.c \
|
||||||
ctype-tis620.c ctype-ujis.c ctype-latin1_de.c \
|
ctype-tis620.c ctype-ujis.c ctype-latin1_de.c \
|
||||||
ctype_autoconf.c \
|
|
||||||
strto.c strings-x86.s \
|
strto.c strings-x86.s \
|
||||||
longlong2str.c longlong2str-x86.s \
|
longlong2str.c longlong2str-x86.s \
|
||||||
strxmov.c bmove_upp.c strappend.c strcont.c strend.c \
|
strxmov.c bmove_upp.c strappend.c strcont.c strend.c \
|
||||||
|
@ -80,7 +79,7 @@ conf_to_src_LDFLAGS= @NOINST_LDFLAGS@
|
||||||
strtoull.o: @CHARSET_OBJS@
|
strtoull.o: @CHARSET_OBJS@
|
||||||
|
|
||||||
clean-local:
|
clean-local:
|
||||||
rm -f ctype_extra_sources.c
|
rm -f ctype_extra_sources.c ctype_autoconf.c
|
||||||
|
|
||||||
if ASSEMBLER
|
if ASSEMBLER
|
||||||
# On Linux gcc can compile the assembly files
|
# On Linux gcc can compile the assembly files
|
||||||
|
|
|
@ -59,6 +59,7 @@ main( int argc,
|
||||||
char* ca_file = 0, *ca_path = 0;
|
char* ca_file = 0, *ca_path = 0;
|
||||||
char* cipher=0;
|
char* cipher=0;
|
||||||
int child_pid,sv[2];
|
int child_pid,sv[2];
|
||||||
|
my_bool unused;
|
||||||
struct st_VioSSLAcceptorFd* ssl_acceptor=0;
|
struct st_VioSSLAcceptorFd* ssl_acceptor=0;
|
||||||
struct st_VioSSLConnectorFd* ssl_connector=0;
|
struct st_VioSSLConnectorFd* ssl_connector=0;
|
||||||
Vio* client_vio=0, *server_vio=0;
|
Vio* client_vio=0, *server_vio=0;
|
||||||
|
@ -96,11 +97,11 @@ main( int argc,
|
||||||
|
|
||||||
client_vio = (struct st_vio*)my_malloc(sizeof(struct st_vio),MYF(0));
|
client_vio = (struct st_vio*)my_malloc(sizeof(struct st_vio),MYF(0));
|
||||||
client_vio->sd = sv[0];
|
client_vio->sd = sv[0];
|
||||||
client_vio->vioblocking(client_vio,0);
|
client_vio->vioblocking(client_vio, 0, &unused);
|
||||||
sslconnect(ssl_connector,client_vio,60L);
|
sslconnect(ssl_connector,client_vio,60L);
|
||||||
server_vio = (struct st_vio*)my_malloc(sizeof(struct st_vio),MYF(0));
|
server_vio = (struct st_vio*)my_malloc(sizeof(struct st_vio),MYF(0));
|
||||||
server_vio->sd = sv[1];
|
server_vio->sd = sv[1];
|
||||||
server_vio->vioblocking(client_vio,0);
|
server_vio->vioblocking(client_vio, 0, &unused);
|
||||||
sslaccept(ssl_acceptor,server_vio,60L);
|
sslaccept(ssl_acceptor,server_vio,60L);
|
||||||
|
|
||||||
printf("Socketpair: %d , %d\n", client_vio->sd, server_vio->sd);
|
printf("Socketpair: %d , %d\n", client_vio->sd, server_vio->sd);
|
||||||
|
|
Loading…
Reference in a new issue