~drizzle-trunk/drizzle/development

« back to all changes in this revision

Viewing changes to plugin/innobase/srv/srv0srv.c

  • Committer: Marisa Plumb
  • Date: 2010-12-04 02:38:29 UTC
  • mto: This revision was merged to the branch mainline in revision 1984.
  • Revision ID: marisa.plumb@gmail.com-20101204023829-2khzxh30wxi256db
updates to a few sql docs 

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
1
/*****************************************************************************
2
2
 
3
 
Copyright (C) 1995, 2010, Innobase Oy. All Rights Reserved.
4
 
Copyright (C) 2008, 2009 Google Inc.
5
 
Copyright (C) 2009, Percona Inc.
 
3
Copyright (c) 1995, 2010, Innobase Oy. All Rights Reserved.
 
4
Copyright (c) 2008, 2009 Google Inc.
 
5
Copyright (c) 2009, Percona Inc.
6
6
 
7
7
Portions of this file contain modifications contributed and copyrighted by
8
8
Google, Inc. Those modifications are gratefully acknowledged and are described
60
60
/* Dummy comment */
61
61
#include "srv0srv.h"
62
62
 
63
 
#include <drizzled/error.h>
64
 
#include <drizzled/errmsg_print.h>
65
 
 
66
63
#include "ut0mem.h"
67
64
#include "ut0ut.h"
68
65
#include "os0proc.h"
71
68
#include "sync0sync.h"
72
69
#include "thr0loc.h"
73
70
#include "que0que.h"
 
71
#include "srv0que.h"
74
72
#include "log0recv.h"
75
73
#include "pars0pars.h"
76
74
#include "usr0sess.h"
109
107
 
110
108
UNIV_INTERN const char* srv_main_thread_op_info = "";
111
109
 
 
110
/** Prefix used by MySQL to indicate pre-5.1 table name encoding */
 
111
UNIV_INTERN const char  srv_mysql50_table_name_prefix[9] = "#mysql50#";
 
112
 
112
113
/* Server parameters which are read from the initfile */
113
114
 
114
115
/* The following three are dir paths which are catenated before file
127
128
/** Whether to check file format during startup.  A value of
128
129
DICT_TF_FORMAT_MAX + 1 means no checking ie. FALSE.  The default is to
129
130
set it to the highest format we support. */
130
 
UNIV_INTERN ulint       srv_max_file_format_at_startup = DICT_TF_FORMAT_MAX;
 
131
UNIV_INTERN ulint       srv_check_file_format_at_startup = DICT_TF_FORMAT_MAX;
131
132
 
132
133
#if DICT_TF_FORMAT_51
133
134
# error "DICT_TF_FORMAT_51 must be 0!"
136
137
on duplicate key checking and foreign key checking */
137
138
UNIV_INTERN ibool       srv_locks_unsafe_for_binlog = FALSE;
138
139
 
139
 
/* If this flag is TRUE, then we will use the native aio of the
140
 
OS (provided we compiled Innobase with it in), otherwise we will
141
 
use simulated aio we build below with threads.
142
 
Currently we support native aio on windows and linux */
143
 
UNIV_INTERN my_bool     srv_use_native_aio = TRUE;
144
 
 
145
 
#ifdef __WIN__
146
 
/* Windows native condition variables. We use runtime loading / function
147
 
pointers, because they are not available on Windows Server 2003 and
148
 
Windows XP/2000.
149
 
 
150
 
We use condition for events on Windows if possible, even if os_event
151
 
resembles Windows kernel event object well API-wise. The reason is
152
 
performance, kernel objects are heavyweights and WaitForSingleObject() is a
153
 
performance killer causing calling thread to context switch. Besides, Innodb
154
 
is preallocating large number (often millions) of os_events. With kernel event
155
 
objects it takes a big chunk out of non-paged pool, which is better suited
156
 
for tasks like IO than for storing idle event objects. */
157
 
UNIV_INTERN ibool       srv_use_native_conditions = FALSE;
158
 
#endif /* __WIN__ */
159
 
 
160
140
UNIV_INTERN ulint       srv_n_data_files = 0;
161
141
UNIV_INTERN char**      srv_data_file_names = NULL;
162
142
/* size in database pages */
204
184
/** The sort order table of the MySQL latin1_swedish_ci character set
205
185
collation */
206
186
#if defined(BUILD_DRIZZLE)
207
 
const byte      srv_latin1_ordering[256]        /* The sort order table of the latin1
 
187
UNIV_INTERN const byte  srv_latin1_ordering[256]        /* The sort order table of the latin1
208
188
                                        character set. The following table is
209
189
                                        the MySQL order as of Feb 10th, 2002 */
210
190
= {
250
230
UNIV_INTERN my_bool     srv_use_sys_malloc      = TRUE;
251
231
/* requested size in kilobytes */
252
232
UNIV_INTERN ulint       srv_buf_pool_size       = ULINT_MAX;
253
 
/* requested number of buffer pool instances */
254
 
UNIV_INTERN ulint       srv_buf_pool_instances  = 1;
255
233
/* previously requested size */
256
234
UNIV_INTERN ulint       srv_buf_pool_old_size;
257
235
/* current size in kilobytes */
303
281
 
304
282
UNIV_INTERN ulong       srv_max_buf_pool_modified_pct   = 75;
305
283
 
306
 
/* the number of purge threads to use from the worker pool (currently 0 or 1).*/
307
 
UNIV_INTERN ulong srv_n_purge_threads = 0;
308
 
 
309
 
/* the number of records to purge in one batch */
310
 
UNIV_INTERN ulong srv_purge_batch_size = 20;
311
 
 
312
284
/* variable counts amount of data read in total (in bytes) */
313
285
UNIV_INTERN ulint srv_data_read = 0;
314
286
 
463
435
UNIV_INTERN ib_int64_t  srv_n_lock_wait_time            = 0;
464
436
UNIV_INTERN ulint               srv_n_lock_max_wait_time        = 0;
465
437
 
466
 
UNIV_INTERN ulint               srv_truncated_status_writes     = 0;
467
438
 
468
439
/*
469
440
  Set the following to 0 if you want InnoDB to write messages on
487
458
 
488
459
/* Mutex for locking srv_monitor_file */
489
460
UNIV_INTERN mutex_t     srv_monitor_file_mutex;
490
 
 
491
 
#ifdef UNIV_PFS_MUTEX
492
 
/* Key to register kernel_mutex with performance schema */
493
 
UNIV_INTERN mysql_pfs_key_t     kernel_mutex_key;
494
 
/* Key to protect writing the commit_id to the sys header */
495
 
UNIV_INTERN mysql_pfs_key_t     commit_id_mutex_key;
496
 
/* Key to register srv_innodb_monitor_mutex with performance schema */
497
 
UNIV_INTERN mysql_pfs_key_t     srv_innodb_monitor_mutex_key;
498
 
/* Key to register srv_monitor_file_mutex with performance schema */
499
 
UNIV_INTERN mysql_pfs_key_t     srv_monitor_file_mutex_key;
500
 
/* Key to register srv_dict_tmpfile_mutex with performance schema */
501
 
UNIV_INTERN mysql_pfs_key_t     srv_dict_tmpfile_mutex_key;
502
 
/* Key to register the mutex with performance schema */
503
 
UNIV_INTERN mysql_pfs_key_t     srv_misc_tmpfile_mutex_key;
504
 
#endif /* UNIV_PFS_MUTEX */
505
 
 
506
461
/* Temporary file for innodb monitor output */
507
462
UNIV_INTERN FILE*       srv_monitor_file;
508
463
/* Mutex for locking srv_dict_tmpfile.
736
691
/* Table for MySQL threads where they will be suspended to wait for locks */
737
692
UNIV_INTERN srv_slot_t* srv_mysql_table = NULL;
738
693
 
739
 
UNIV_INTERN os_event_t  srv_timeout_event;
740
 
 
741
 
UNIV_INTERN os_event_t  srv_monitor_event;
742
 
 
743
 
UNIV_INTERN os_event_t  srv_error_event;
744
 
 
745
694
UNIV_INTERN os_event_t  srv_lock_timeout_thread_event;
746
695
 
747
696
UNIV_INTERN srv_sys_t*  srv_sys = NULL;
751
700
UNIV_INTERN byte        srv_pad1[64];
752
701
/* mutex protecting the server, trx structs, query threads, and lock table */
753
702
UNIV_INTERN mutex_t*    kernel_mutex_temp;
754
 
/* mutex protecting the sys header for writing the commit id */
755
 
UNIV_INTERN mutex_t*    commit_id_mutex_temp;
756
 
 
757
703
/* padding to prevent other memory update hotspots from residing on
758
704
the same memory cache line */
759
705
UNIV_INTERN byte        srv_pad2[64];
772
718
static ulint    srv_meter_foreground[SRV_MASTER + 1];
773
719
#endif
774
720
 
 
721
/* The following values give info about the activity going on in
 
722
the database. They are protected by the server mutex. The arrays
 
723
are indexed by the type of the thread. */
 
724
 
 
725
UNIV_INTERN ulint       srv_n_threads_active[SRV_MASTER + 1];
 
726
UNIV_INTERN ulint       srv_n_threads[SRV_MASTER + 1];
 
727
 
775
728
/***********************************************************************
776
729
Prints counters for work done by srv_master_thread. */
777
730
static
789
742
                      srv_log_writes_and_flush);
790
743
}
791
744
 
792
 
/* The following values give info about the activity going on in
793
 
the database. They are protected by the server mutex. The arrays
794
 
are indexed by the type of the thread. */
795
 
 
796
 
UNIV_INTERN ulint       srv_n_threads_active[SRV_MASTER + 1];
797
 
UNIV_INTERN ulint       srv_n_threads[SRV_MASTER + 1];
798
 
 
799
745
/*********************************************************************//**
800
746
Sets the info describing an i/o thread current state. */
801
747
UNIV_INTERN
915
861
 
916
862
        slot = srv_table_get_nth_slot(slot_no);
917
863
 
918
 
        type = static_cast<srv_thread_type>(slot->type);
 
864
        type = slot->type;
919
865
 
920
866
        ut_ad(type >= SRV_WORKER);
921
867
        ut_ad(type <= SRV_MASTER);
958
904
 
959
905
                slot = srv_table_get_nth_slot(i);
960
906
 
961
 
                if (slot->in_use &&
962
 
                    (static_cast<srv_thread_type>(slot->type) == type) &&
963
 
                    slot->suspended) {
 
907
                if (slot->in_use && slot->type == type && slot->suspended) {
964
908
 
965
909
                        slot->suspended = FALSE;
966
910
 
1005
949
 
1006
950
        slot = srv_table_get_nth_slot(slot_no);
1007
951
 
1008
 
        type = static_cast<srv_thread_type>(slot->type);
 
952
        type = slot->type;
1009
953
 
1010
954
        ut_ad(type >= SRV_WORKER);
1011
955
        ut_ad(type <= SRV_MASTER);
1026
970
        srv_slot_t*             slot;
1027
971
        ulint                   i;
1028
972
 
1029
 
        srv_sys = static_cast<srv_sys_t *>(mem_alloc(sizeof(srv_sys_t)));
1030
 
 
1031
 
        kernel_mutex_temp = static_cast<ib_mutex_t *>(mem_alloc(sizeof(mutex_t)));
1032
 
        mutex_create(kernel_mutex_key, &kernel_mutex, SYNC_KERNEL);
1033
 
 
1034
 
        commit_id_mutex_temp = static_cast<ib_mutex_t *>(mem_alloc(sizeof(mutex_t)));
1035
 
        mutex_create(commit_id_mutex_key, &commit_id_mutex, SYNC_COMMIT_ID_LOCK);
1036
 
 
1037
 
        mutex_create(srv_innodb_monitor_mutex_key,
1038
 
                     &srv_innodb_monitor_mutex, SYNC_NO_ORDER_CHECK);
1039
 
 
1040
 
        srv_sys->threads = static_cast<srv_table_t *>(mem_alloc(OS_THREAD_MAX_N * sizeof(srv_slot_t)));
 
973
        srv_sys = mem_alloc(sizeof(srv_sys_t));
 
974
 
 
975
        kernel_mutex_temp = mem_alloc(sizeof(mutex_t));
 
976
        mutex_create(&kernel_mutex, SYNC_KERNEL);
 
977
 
 
978
        mutex_create(&srv_innodb_monitor_mutex, SYNC_NO_ORDER_CHECK);
 
979
 
 
980
        srv_sys->threads = mem_alloc(OS_THREAD_MAX_N * sizeof(srv_slot_t));
1041
981
 
1042
982
        for (i = 0; i < OS_THREAD_MAX_N; i++) {
1043
983
                slot = srv_table_get_nth_slot(i);
1047
987
                ut_a(slot->event);
1048
988
        }
1049
989
 
1050
 
        srv_mysql_table = static_cast<srv_slot_t *>(mem_alloc(OS_THREAD_MAX_N * sizeof(srv_slot_t)));
 
990
        srv_mysql_table = mem_alloc(OS_THREAD_MAX_N * sizeof(srv_slot_t));
1051
991
 
1052
992
        for (i = 0; i < OS_THREAD_MAX_N; i++) {
1053
993
                slot = srv_mysql_table + i;
1057
997
                ut_a(slot->event);
1058
998
        }
1059
999
 
1060
 
        srv_error_event = os_event_create(NULL);
1061
 
 
1062
 
        srv_timeout_event = os_event_create(NULL);
1063
 
 
1064
 
        srv_monitor_event = os_event_create(NULL);
1065
 
 
1066
1000
        srv_lock_timeout_thread_event = os_event_create(NULL);
1067
1001
 
1068
1002
        for (i = 0; i < SRV_MASTER + 1; i++) {
1089
1023
 
1090
1024
        UT_LIST_INIT(srv_conc_queue);
1091
1025
 
1092
 
        srv_conc_slots = static_cast<srv_conc_slot_t *>(mem_alloc(OS_THREAD_MAX_N * sizeof(srv_conc_slot_t)));
 
1026
        srv_conc_slots = mem_alloc(OS_THREAD_MAX_N * sizeof(srv_conc_slot_t));
1093
1027
 
1094
1028
        for (i = 0; i < OS_THREAD_MAX_N; i++) {
1095
1029
                conc_slot = srv_conc_slots + i;
1122
1056
        mem_free(srv_mysql_table);
1123
1057
        srv_mysql_table = NULL;
1124
1058
 
1125
 
        mem_free(commit_id_mutex_temp);
1126
 
        commit_id_mutex_temp = NULL;
1127
 
 
1128
1059
        trx_i_s_cache_free(trx_i_s_cache);
1129
1060
}
1130
1061
 
1645
1576
                row_mysql_unfreeze_data_dictionary(trx);
1646
1577
                break;
1647
1578
        case RW_X_LATCH:
1648
 
                /* There should never be a lock wait when the
1649
 
                dictionary latch is reserved in X mode.  Dictionary
1650
 
                transactions should only acquire locks on dictionary
1651
 
                tables, not other tables. All access to dictionary
1652
 
                tables should be covered by dictionary
1653
 
                transactions. */
1654
 
                ut_print_timestamp(stderr);
1655
 
                fputs("  InnoDB: Error: dict X latch held in "
1656
 
                      "srv_suspend_mysql_thread\n", stderr);
1657
 
                /* This should never occur. This incorrect handling
1658
 
                was added in the early development of
1659
 
                ha_innobase::add_index() in InnoDB Plugin 1.0. */
1660
1579
                /* Release fast index creation latch */
1661
1580
                row_mysql_unlock_data_dictionary(trx);
1662
1581
                break;
1676
1595
                row_mysql_freeze_data_dictionary(trx);
1677
1596
                break;
1678
1597
        case RW_X_LATCH:
1679
 
                /* This should never occur. This incorrect handling
1680
 
                was added in the early development of
1681
 
                ha_innobase::add_index() in InnoDB Plugin 1.0. */
1682
1598
                row_mysql_lock_data_dictionary(trx);
1683
1599
                break;
1684
1600
        }
1715
1631
                    start_time != -1 && finish_time != -1) {
1716
1632
                        srv_n_lock_max_wait_time = diff_time;
1717
1633
                }
1718
 
 
1719
 
                /* Record the lock wait time for this thread */
1720
 
                thd_set_lock_wait_time(trx->mysql_thd, diff_time);
1721
1634
        }
1722
1635
 
1723
1636
        if (trx->was_chosen_as_deadlock_victim) {
1795
1708
 
1796
1709
        log_refresh_stats();
1797
1710
 
1798
 
        buf_refresh_io_stats_all();
 
1711
        buf_refresh_io_stats();
1799
1712
 
1800
1713
        srv_n_rows_inserted_old = srv_n_rows_inserted;
1801
1714
        srv_n_rows_updated_old = srv_n_rows_updated;
1945
1858
                (ulong) srv_conc_n_waiting_threads);
1946
1859
 
1947
1860
        fprintf(file, "%lu read views open inside InnoDB\n",
1948
 
                static_cast<ulint>(UT_LIST_GET_LEN(trx_sys->view_list)));
 
1861
                UT_LIST_GET_LEN(trx_sys->view_list));
1949
1862
 
1950
1863
        n_reserved = fil_space_get_n_reserved_extents(0);
1951
1864
        if (n_reserved > 0) {
2006
1919
srv_export_innodb_status(void)
2007
1920
/*==========================*/
2008
1921
{
2009
 
        buf_pool_stat_t stat;
2010
 
        ulint           LRU_len;
2011
 
        ulint           free_len;
2012
 
        ulint           flush_list_len;
2013
 
 
2014
 
        buf_get_total_stat(&stat);
2015
 
        buf_get_total_list_len(&LRU_len, &free_len, &flush_list_len);
2016
 
 
2017
1922
        mutex_enter(&srv_innodb_monitor_mutex);
2018
1923
 
2019
1924
        export_vars.innodb_data_pending_reads
2028
1933
        export_vars.innodb_data_reads = os_n_file_reads;
2029
1934
        export_vars.innodb_data_writes = os_n_file_writes;
2030
1935
        export_vars.innodb_data_written = srv_data_written;
2031
 
        export_vars.innodb_buffer_pool_read_requests = stat.n_page_gets;
 
1936
        export_vars.innodb_buffer_pool_read_requests = buf_pool->stat.n_page_gets;
2032
1937
        export_vars.innodb_buffer_pool_write_requests
2033
1938
                = srv_buf_pool_write_requests;
2034
1939
        export_vars.innodb_buffer_pool_wait_free = srv_buf_pool_wait_free;
2035
1940
        export_vars.innodb_buffer_pool_pages_flushed = srv_buf_pool_flushed;
2036
1941
        export_vars.innodb_buffer_pool_reads = srv_buf_pool_reads;
2037
1942
        export_vars.innodb_buffer_pool_read_ahead
2038
 
                = stat.n_ra_pages_read;
 
1943
                = buf_pool->stat.n_ra_pages_read;
2039
1944
        export_vars.innodb_buffer_pool_read_ahead_evicted
2040
 
                = stat.n_ra_pages_evicted;
2041
 
        export_vars.innodb_buffer_pool_pages_data = LRU_len;
2042
 
        export_vars.innodb_buffer_pool_pages_dirty = flush_list_len;
2043
 
        export_vars.innodb_buffer_pool_pages_free = free_len;
 
1945
                = buf_pool->stat.n_ra_pages_evicted;
 
1946
        export_vars.innodb_buffer_pool_pages_data
 
1947
                = UT_LIST_GET_LEN(buf_pool->LRU);
 
1948
        export_vars.innodb_buffer_pool_pages_dirty
 
1949
                = UT_LIST_GET_LEN(buf_pool->flush_list);
 
1950
        export_vars.innodb_buffer_pool_pages_free
 
1951
                = UT_LIST_GET_LEN(buf_pool->free);
2044
1952
#ifdef UNIV_DEBUG
2045
1953
        export_vars.innodb_buffer_pool_pages_latched
2046
1954
                = buf_get_latched_pages_number();
2047
1955
#endif /* UNIV_DEBUG */
2048
 
        export_vars.innodb_buffer_pool_pages_total = buf_pool_get_n_pages();
 
1956
        export_vars.innodb_buffer_pool_pages_total = buf_pool->curr_size;
2049
1957
 
2050
 
        export_vars.innodb_buffer_pool_pages_misc
2051
 
                = buf_pool_get_n_pages() - LRU_len - free_len;
 
1958
        export_vars.innodb_buffer_pool_pages_misc = buf_pool->curr_size
 
1959
                - UT_LIST_GET_LEN(buf_pool->LRU)
 
1960
                - UT_LIST_GET_LEN(buf_pool->free);
2052
1961
#ifdef HAVE_ATOMIC_BUILTINS
2053
1962
        export_vars.innodb_have_atomic_builtins = 1;
2054
1963
#else
2064
1973
        export_vars.innodb_log_writes = srv_log_writes;
2065
1974
        export_vars.innodb_dblwr_pages_written = srv_dblwr_pages_written;
2066
1975
        export_vars.innodb_dblwr_writes = srv_dblwr_writes;
2067
 
        export_vars.innodb_pages_created = stat.n_pages_created;
2068
 
        export_vars.innodb_pages_read = stat.n_pages_read;
2069
 
        export_vars.innodb_pages_written = stat.n_pages_written;
 
1976
        export_vars.innodb_pages_created = buf_pool->stat.n_pages_created;
 
1977
        export_vars.innodb_pages_read = buf_pool->stat.n_pages_read;
 
1978
        export_vars.innodb_pages_written = buf_pool->stat.n_pages_written;
2070
1979
        export_vars.innodb_row_lock_waits = srv_n_lock_wait_count;
2071
1980
        export_vars.innodb_row_lock_current_waits
2072
1981
                = srv_n_lock_wait_current_count;
2083
1992
        export_vars.innodb_rows_inserted = srv_n_rows_inserted;
2084
1993
        export_vars.innodb_rows_updated = srv_n_rows_updated;
2085
1994
        export_vars.innodb_rows_deleted = srv_n_rows_deleted;
2086
 
        export_vars.innodb_truncated_status_writes = srv_truncated_status_writes;
2087
1995
 
2088
1996
        mutex_exit(&srv_innodb_monitor_mutex);
2089
1997
}
2095
2003
os_thread_ret_t
2096
2004
srv_monitor_thread(
2097
2005
/*===============*/
2098
 
        void*   /*arg __attribute__((unused))*/)
 
2006
        void*   arg __attribute__((unused)))
2099
2007
                        /*!< in: a dummy parameter required by
2100
2008
                        os_thread_create */
2101
2009
{
2102
 
        ib_int64_t      sig_count;
2103
2010
        double          time_elapsed;
2104
2011
        time_t          current_time;
2105
2012
        time_t          last_table_monitor_time;
2112
2019
        fprintf(stderr, "Lock timeout thread starts, id %lu\n",
2113
2020
                os_thread_pf(os_thread_get_curr_id()));
2114
2021
#endif
2115
 
 
2116
 
#ifdef UNIV_PFS_THREAD
2117
 
        pfs_register_thread(srv_monitor_thread_key);
2118
 
#endif
2119
 
 
2120
 
        srv_last_monitor_time = ut_time();
2121
 
        last_table_monitor_time = ut_time();
2122
 
        last_tablespace_monitor_time = ut_time();
2123
 
        last_monitor_time = ut_time();
 
2022
        UT_NOT_USED(arg);
 
2023
        srv_last_monitor_time = time(NULL);
 
2024
        last_table_monitor_time = time(NULL);
 
2025
        last_tablespace_monitor_time = time(NULL);
 
2026
        last_monitor_time = time(NULL);
2124
2027
        mutex_skipped = 0;
2125
2028
        last_srv_print_monitor = srv_print_innodb_monitor;
2126
2029
loop:
2127
2030
        srv_monitor_active = TRUE;
2128
2031
 
2129
2032
        /* Wake up every 5 seconds to see if we need to print
2130
 
        monitor information or if signalled at shutdown. */
2131
 
 
2132
 
        sig_count = os_event_reset(srv_monitor_event);
2133
 
 
2134
 
        os_event_wait_time_low(srv_monitor_event, 5000000, sig_count);
2135
 
 
2136
 
        current_time = ut_time();
 
2033
        monitor information. */
 
2034
 
 
2035
        os_thread_sleep(5000000);
 
2036
 
 
2037
        current_time = time(NULL);
2137
2038
 
2138
2039
        time_elapsed = difftime(current_time, last_monitor_time);
2139
2040
 
2140
2041
        if (time_elapsed > 15) {
2141
 
                last_monitor_time = ut_time();
 
2042
                last_monitor_time = time(NULL);
2142
2043
 
2143
2044
                if (srv_print_innodb_monitor) {
2144
2045
                        /* Reset mutex_skipped counter everytime
2182
2083
                if (srv_print_innodb_tablespace_monitor
2183
2084
                    && difftime(current_time,
2184
2085
                                last_tablespace_monitor_time) > 60) {
2185
 
                        last_tablespace_monitor_time = ut_time();
 
2086
                        last_tablespace_monitor_time = time(NULL);
2186
2087
 
2187
2088
                        fputs("========================"
2188
2089
                              "========================\n",
2208
2109
                if (srv_print_innodb_table_monitor
2209
2110
                    && difftime(current_time, last_table_monitor_time) > 60) {
2210
2111
 
2211
 
                        last_table_monitor_time = ut_time();
 
2112
                        last_table_monitor_time = time(NULL);
2212
2113
 
2213
2114
                        fputs("===========================================\n",
2214
2115
                              stderr);
2260
2161
os_thread_ret_t
2261
2162
srv_lock_timeout_thread(
2262
2163
/*====================*/
2263
 
        void*   /*arg __attribute__((unused))*/)
 
2164
        void*   arg __attribute__((unused)))
2264
2165
                        /* in: a dummy parameter required by
2265
2166
                        os_thread_create */
2266
2167
{
2268
2169
        ibool           some_waits;
2269
2170
        double          wait_time;
2270
2171
        ulint           i;
2271
 
        ib_int64_t      sig_count;
2272
 
 
2273
 
#ifdef UNIV_PFS_THREAD
2274
 
        pfs_register_thread(srv_lock_timeout_thread_key);
2275
 
#endif
2276
2172
 
2277
2173
loop:
2278
 
 
2279
2174
        /* When someone is waiting for a lock, we wake up every second
2280
2175
        and check if a timeout has passed for a lock wait */
2281
2176
 
2282
 
        sig_count = os_event_reset(srv_timeout_event);
2283
 
 
2284
 
        os_event_wait_time_low(srv_timeout_event, 1000000, sig_count);
 
2177
        os_thread_sleep(1000000);
2285
2178
 
2286
2179
        srv_lock_timeout_active = TRUE;
2287
2180
 
2368
2261
os_thread_ret_t
2369
2262
srv_error_monitor_thread(
2370
2263
/*=====================*/
2371
 
        void*   /*arg __attribute__((unused))*/)
 
2264
        void*   arg __attribute__((unused)))
2372
2265
                        /*!< in: a dummy parameter required by
2373
2266
                        os_thread_create */
2374
2267
{
2376
2269
        ulint           fatal_cnt       = 0;
2377
2270
        ib_uint64_t     old_lsn;
2378
2271
        ib_uint64_t     new_lsn;
2379
 
        ib_int64_t      sig_count;
2380
2272
 
2381
2273
        old_lsn = srv_start_lsn;
2382
2274
 
2384
2276
        fprintf(stderr, "Error monitor thread starts, id %lu\n",
2385
2277
                os_thread_pf(os_thread_get_curr_id()));
2386
2278
#endif
2387
 
 
2388
 
#ifdef UNIV_PFS_THREAD
2389
 
        pfs_register_thread(srv_error_monitor_thread_key);
2390
 
#endif
2391
 
 
2392
2279
loop:
2393
2280
        srv_error_monitor_active = TRUE;
2394
2281
 
2398
2285
        new_lsn = log_get_lsn();
2399
2286
 
2400
2287
        if (new_lsn < old_lsn) {
2401
 
          drizzled::errmsg_printf(drizzled::error::INFO,
2402
 
                                  "InnoDB: Error: old log sequence number %"PRIu64" was greater than the new log sequence number %"PRIu64"!"
2403
 
                                  "InnoDB: Please submit a bug report to http://bugs.launchpad.net/drizzle",
2404
 
                                  old_lsn, new_lsn);
 
2288
                ut_print_timestamp(stderr);
 
2289
                fprintf(stderr,
 
2290
                        "  InnoDB: Error: old log sequence number %"PRIu64""
 
2291
                        " was greater\n"
 
2292
                        "InnoDB: than the new log sequence number %"PRIu64"!\n"
 
2293
                        "InnoDB: Please submit a bug report"
 
2294
                        " to http://bugs.mysql.com\n",
 
2295
                        old_lsn, new_lsn);
2405
2296
        }
2406
2297
 
2407
2298
        old_lsn = new_lsn;
2448
2339
 
2449
2340
        fflush(stderr);
2450
2341
 
2451
 
        sig_count = os_event_reset(srv_error_event);
2452
 
 
2453
 
        os_event_wait_time_low(srv_error_event, 1000000, sig_count);
 
2342
        os_thread_sleep(1000000);
2454
2343
 
2455
2344
        if (srv_shutdown_state < SRV_SHUTDOWN_CLEANUP) {
2456
2345
 
2467
2356
        OS_THREAD_DUMMY_RETURN;
2468
2357
}
2469
2358
 
2470
 
/**********************************************************************//**
2471
 
Check whether any background thread is active.
2472
 
@return FALSE if all are are suspended or have exited. */
2473
 
UNIV_INTERN
2474
 
ibool
2475
 
srv_is_any_background_thread_active(void)
2476
 
/*=====================================*/
2477
 
{
2478
 
        ulint   i;
2479
 
        ibool   ret = FALSE;
2480
 
 
2481
 
        mutex_enter(&kernel_mutex);
2482
 
 
2483
 
        for (i = SRV_COM; i <= SRV_MASTER; ++i) {
2484
 
                if (srv_n_threads_active[i] != 0) {
2485
 
                        ret = TRUE;
2486
 
                        break;
2487
 
                }
2488
 
        }
2489
 
 
2490
 
        mutex_exit(&kernel_mutex);
2491
 
 
2492
 
        return(ret);
2493
 
}
2494
 
 
2495
2359
/*******************************************************************//**
2496
2360
Tells the InnoDB server that there has been activity in the database
2497
2361
and wakes up the master thread if it is suspended (not sleeping). Used
2498
2362
in the MySQL interface. Note that there is a small chance that the master
2499
 
thread stays suspended (we do not protect our operation with the
2500
 
srv_sys_t->mutex, for performance reasons). */
 
2363
thread stays suspended (we do not protect our operation with the kernel
 
2364
mutex, for performace reasons). */
2501
2365
UNIV_INTERN
2502
2366
void
2503
2367
srv_active_wake_master_thread(void)
2516
2380
}
2517
2381
 
2518
2382
/*******************************************************************//**
2519
 
Tells the purge thread that there has been activity in the database
2520
 
and wakes up the purge thread if it is suspended (not sleeping).  Note
2521
 
that there is a small chance that the purge thread stays suspended
2522
 
(we do not protect our operation with the kernel mutex, for
2523
 
performace reasons). */
2524
 
UNIV_INTERN
2525
 
void
2526
 
srv_wake_purge_thread_if_not_active(void)
2527
 
/*=====================================*/
2528
 
{
2529
 
        ut_ad(!mutex_own(&kernel_mutex));
2530
 
 
2531
 
        if (srv_n_purge_threads > 0
2532
 
            && srv_n_threads_active[SRV_WORKER] == 0) {
2533
 
 
2534
 
                mutex_enter(&kernel_mutex);
2535
 
 
2536
 
                srv_release_threads(SRV_WORKER, 1);
2537
 
 
2538
 
                mutex_exit(&kernel_mutex);
2539
 
        }
2540
 
}
2541
 
 
2542
 
/*******************************************************************//**
2543
2383
Wakes up the master thread if it is suspended or being suspended. */
2544
2384
UNIV_INTERN
2545
2385
void
2555
2395
        mutex_exit(&kernel_mutex);
2556
2396
}
2557
2397
 
2558
 
/*******************************************************************//**
2559
 
Wakes up the purge thread if it's not already awake. */
2560
 
UNIV_INTERN
2561
 
void
2562
 
srv_wake_purge_thread(void)
2563
 
/*=======================*/
2564
 
{
2565
 
        ut_ad(!mutex_own(&kernel_mutex));
2566
 
 
2567
 
        if (srv_n_purge_threads > 0) {
2568
 
 
2569
 
                mutex_enter(&kernel_mutex);
2570
 
 
2571
 
                srv_release_threads(SRV_WORKER, 1);
2572
 
 
2573
 
                mutex_exit(&kernel_mutex);
2574
 
        }
2575
 
}
2576
 
 
2577
2398
/**********************************************************************
2578
2399
The master thread is tasked to ensure that flush of log file happens
2579
2400
once every second in the background. This is to ensure that not more
2594
2415
        }
2595
2416
}
2596
2417
 
2597
 
/********************************************************************//**
2598
 
Do a full purge, reconfigure the purge sub-system if a dynamic
2599
 
change is detected. */
2600
 
static
2601
 
void
2602
 
srv_master_do_purge(void)
2603
 
/*=====================*/
2604
 
{
2605
 
        ulint   n_pages_purged;
2606
 
 
2607
 
        ut_ad(!mutex_own(&kernel_mutex));
2608
 
 
2609
 
        ut_a(srv_n_purge_threads == 0);
2610
 
 
2611
 
        do {
2612
 
                /* Check for shutdown and change in purge config. */
2613
 
                if (srv_fast_shutdown && srv_shutdown_state > 0) {
2614
 
                        /* Nothing to purge. */
2615
 
                        n_pages_purged = 0;
2616
 
                } else {
2617
 
                        n_pages_purged = trx_purge(srv_purge_batch_size);
2618
 
                }
2619
 
 
2620
 
                srv_sync_log_buffer_in_background();
2621
 
 
2622
 
        } while (n_pages_purged > 0);
2623
 
}
2624
 
 
2625
2418
/*********************************************************************//**
2626
2419
The master thread controlling the server.
2627
2420
@return a dummy parameter */
2629
2422
os_thread_ret_t
2630
2423
srv_master_thread(
2631
2424
/*==============*/
2632
 
        void*   /*arg __attribute__((unused))*/)
 
2425
        void*   arg __attribute__((unused)))
2633
2426
                        /*!< in: a dummy parameter required by
2634
2427
                        os_thread_create */
2635
2428
{
2636
 
        buf_pool_stat_t buf_stat;
2637
2429
        os_event_t      event;
2638
2430
        ulint           old_activity_count;
2639
2431
        ulint           n_pages_purged  = 0;
2645
2437
        ulint           n_ios_old;
2646
2438
        ulint           n_ios_very_old;
2647
2439
        ulint           n_pend_ios;
2648
 
        ulint           next_itr_time;
 
2440
        ibool           skip_sleep      = FALSE;
2649
2441
        ulint           i;
2650
2442
 
2651
2443
#ifdef UNIV_DEBUG_THREAD_CREATION
2652
2444
        fprintf(stderr, "Master thread starts, id %lu\n",
2653
2445
                os_thread_pf(os_thread_get_curr_id()));
2654
2446
#endif
2655
 
 
2656
 
#ifdef UNIV_PFS_THREAD
2657
 
        pfs_register_thread(srv_master_thread_key);
2658
 
#endif
2659
 
 
2660
2447
        srv_main_thread_process_no = os_proc_get_number();
2661
2448
        srv_main_thread_id = os_thread_pf(os_thread_get_curr_id());
2662
2449
 
2675
2462
 
2676
2463
        srv_main_thread_op_info = "reserving kernel mutex";
2677
2464
 
2678
 
        buf_get_total_stat(&buf_stat);
2679
 
        n_ios_very_old = log_sys->n_log_ios + buf_stat.n_pages_read
2680
 
                + buf_stat.n_pages_written;
 
2465
        n_ios_very_old = log_sys->n_log_ios + buf_pool->stat.n_pages_read
 
2466
                + buf_pool->stat.n_pages_written;
2681
2467
        mutex_enter(&kernel_mutex);
2682
2468
 
2683
2469
        /* Store the user activity counter at the start of this loop */
2694
2480
        when there is database activity */
2695
2481
 
2696
2482
        srv_last_log_flush_time = time(NULL);
2697
 
 
2698
 
        /* Sleep for 1 second on entrying the for loop below the first time. */
2699
 
        next_itr_time = ut_time_ms() + 1000;
 
2483
        skip_sleep = FALSE;
2700
2484
 
2701
2485
        for (i = 0; i < 10; i++) {
2702
 
                ulint   cur_time = ut_time_ms();
 
2486
                n_ios_old = log_sys->n_log_ios + buf_pool->stat.n_pages_read
 
2487
                        + buf_pool->stat.n_pages_written;
 
2488
                srv_main_thread_op_info = "sleeping";
 
2489
                srv_main_1_second_loops++;
 
2490
 
 
2491
                if (!skip_sleep) {
 
2492
 
 
2493
                        os_thread_sleep(1000000);
 
2494
                        srv_main_sleeps++;
 
2495
                }
 
2496
 
 
2497
                skip_sleep = FALSE;
2703
2498
 
2704
2499
                /* ALTER TABLE in MySQL requires on Unix that the table handler
2705
2500
                can drop tables lazily after there no longer are SELECT
2716
2511
                        goto background_loop;
2717
2512
                }
2718
2513
 
2719
 
                buf_get_total_stat(&buf_stat);
2720
 
 
2721
 
                n_ios_old = log_sys->n_log_ios + buf_stat.n_pages_read
2722
 
                        + buf_stat.n_pages_written;
2723
 
 
2724
 
                srv_main_thread_op_info = "sleeping";
2725
 
                srv_main_1_second_loops++;
2726
 
 
2727
 
                if (next_itr_time > cur_time
2728
 
                    && srv_shutdown_state == SRV_SHUTDOWN_NONE) {
2729
 
 
2730
 
                        /* Get sleep interval in micro seconds. We use
2731
 
                        ut_min() to avoid long sleep in case of
2732
 
                        wrap around. */
2733
 
                        os_thread_sleep(ut_min(1000000,
2734
 
                                        (next_itr_time - cur_time)
2735
 
                                         * 1000));
2736
 
                        srv_main_sleeps++;
2737
 
                }
2738
 
 
2739
 
                /* Each iteration should happen at 1 second interval. */
2740
 
                next_itr_time = ut_time_ms() + 1000;
2741
 
 
2742
2514
                /* Flush logs if needed */
2743
2515
                srv_sync_log_buffer_in_background();
2744
2516
 
2746
2518
                log_free_check();
2747
2519
 
2748
2520
                /* If i/os during one second sleep were less than 5% of
2749
 
                capacity, we assume that there is free disk i/o capacity
2750
 
                available, and it makes sense to do an insert buffer merge. */
 
2521
                capacity, we assume that there is free disk i/o capacity
 
2522
                available, and it makes sense to do an insert buffer merge. */
2751
2523
 
2752
 
                buf_get_total_stat(&buf_stat);
2753
2524
                n_pend_ios = buf_get_n_pending_ios()
2754
2525
                        + log_sys->n_pending_writes;
2755
 
                n_ios = log_sys->n_log_ios + buf_stat.n_pages_read
2756
 
                        + buf_stat.n_pages_written;
 
2526
                n_ios = log_sys->n_log_ios + buf_pool->stat.n_pages_read
 
2527
                        + buf_pool->stat.n_pages_written;
2757
2528
                if (n_pend_ios < SRV_PEND_IO_THRESHOLD
2758
2529
                    && (n_ios - n_ios_old < SRV_RECENT_IO_ACTIVITY)) {
2759
2530
                        srv_main_thread_op_info = "doing insert buffer merge";
2771
2542
 
2772
2543
                        srv_main_thread_op_info =
2773
2544
                                "flushing buffer pool pages";
2774
 
                        n_pages_flushed = buf_flush_list(
2775
 
                                PCT_IO(100), IB_ULONGLONG_MAX);
2776
 
 
 
2545
                        n_pages_flushed = buf_flush_batch(BUF_FLUSH_LIST,
 
2546
                                                          PCT_IO(100),
 
2547
                                                          IB_ULONGLONG_MAX);
 
2548
 
 
2549
                        /* If we had to do the flush, it may have taken
 
2550
                        even more than 1 second, and also, there may be more
 
2551
                        to flush. Do not sleep 1 second during the next
 
2552
                        iteration of this loop. */
 
2553
 
 
2554
                        skip_sleep = TRUE;
2777
2555
                } else if (srv_adaptive_flushing) {
2778
2556
 
2779
2557
                        /* Try to keep the rate of flushing of dirty
2786
2564
                                        "flushing buffer pool pages";
2787
2565
                                n_flush = ut_min(PCT_IO(100), n_flush);
2788
2566
                                n_pages_flushed =
2789
 
                                        buf_flush_list(
 
2567
                                        buf_flush_batch(
 
2568
                                                BUF_FLUSH_LIST,
2790
2569
                                                n_flush,
2791
2570
                                                IB_ULONGLONG_MAX);
 
2571
 
 
2572
                                if (n_flush == PCT_IO(100)) {
 
2573
                                        skip_sleep = TRUE;
 
2574
                                }
2792
2575
                        }
2793
2576
                }
2794
2577
 
2818
2601
        loop above requests writes for that case. The writes done here
2819
2602
        are not required, and may be disabled. */
2820
2603
 
2821
 
        buf_get_total_stat(&buf_stat);
2822
2604
        n_pend_ios = buf_get_n_pending_ios() + log_sys->n_pending_writes;
2823
 
        n_ios = log_sys->n_log_ios + buf_stat.n_pages_read
2824
 
                + buf_stat.n_pages_written;
 
2605
        n_ios = log_sys->n_log_ios + buf_pool->stat.n_pages_read
 
2606
                + buf_pool->stat.n_pages_written;
2825
2607
 
2826
2608
        srv_main_10_second_loops++;
2827
2609
        if (n_pend_ios < SRV_PEND_IO_THRESHOLD
2828
2610
            && (n_ios - n_ios_very_old < SRV_PAST_IO_ACTIVITY)) {
2829
2611
 
2830
2612
                srv_main_thread_op_info = "flushing buffer pool pages";
2831
 
                buf_flush_list(PCT_IO(100), IB_ULONGLONG_MAX);
 
2613
                buf_flush_batch(BUF_FLUSH_LIST, PCT_IO(100),
 
2614
                                IB_ULONGLONG_MAX);
2832
2615
 
2833
2616
                /* Flush logs if needed */
2834
2617
                srv_sync_log_buffer_in_background();
2843
2626
        /* Flush logs if needed */
2844
2627
        srv_sync_log_buffer_in_background();
2845
2628
 
2846
 
        if (srv_n_purge_threads == 0) {
2847
 
                srv_main_thread_op_info = "master purging";
2848
 
 
2849
 
                srv_master_do_purge();
 
2629
        /* We run a full purge every 10 seconds, even if the server
 
2630
        were active */
 
2631
        do {
2850
2632
 
2851
2633
                if (srv_fast_shutdown && srv_shutdown_state > 0) {
2852
2634
 
2853
2635
                        goto background_loop;
2854
2636
                }
2855
 
        }
 
2637
 
 
2638
                srv_main_thread_op_info = "purging";
 
2639
                n_pages_purged = trx_purge();
 
2640
 
 
2641
                /* Flush logs if needed */
 
2642
                srv_sync_log_buffer_in_background();
 
2643
 
 
2644
        } while (n_pages_purged);
2856
2645
 
2857
2646
        srv_main_thread_op_info = "flushing buffer pool pages";
2858
2647
 
2864
2653
                (> 70 %), we assume we can afford reserving the disk(s) for
2865
2654
                the time it requires to flush 100 pages */
2866
2655
 
2867
 
                n_pages_flushed = buf_flush_list(
2868
 
                        PCT_IO(100), IB_ULONGLONG_MAX);
 
2656
                n_pages_flushed = buf_flush_batch(BUF_FLUSH_LIST,
 
2657
                                                  PCT_IO(100),
 
2658
                                                  IB_ULONGLONG_MAX);
2869
2659
        } else {
2870
2660
                /* Otherwise, we only flush a small number of pages so that
2871
2661
                we do not unnecessarily use much disk i/o capacity from
2872
2662
                other work */
2873
2663
 
2874
 
                n_pages_flushed = buf_flush_list(
2875
 
                          PCT_IO(10), IB_ULONGLONG_MAX);
 
2664
                n_pages_flushed = buf_flush_batch(BUF_FLUSH_LIST,
 
2665
                                                  PCT_IO(10),
 
2666
                                                  IB_ULONGLONG_MAX);
2876
2667
        }
2877
2668
 
2878
2669
        srv_main_thread_op_info = "making checkpoint";
2916
2707
                MySQL tries to drop a table while there are still open handles
2917
2708
                to it and we had to put it to the background drop queue.) */
2918
2709
 
2919
 
                if (srv_shutdown_state == SRV_SHUTDOWN_NONE) {
2920
 
                        os_thread_sleep(100000);
 
2710
                os_thread_sleep(100000);
 
2711
        }
 
2712
 
 
2713
        srv_main_thread_op_info = "purging";
 
2714
 
 
2715
        /* Run a full purge */
 
2716
        do {
 
2717
                if (srv_fast_shutdown && srv_shutdown_state > 0) {
 
2718
 
 
2719
                        break;
2921
2720
                }
2922
 
        }
2923
 
 
2924
 
        if (srv_n_purge_threads == 0) {
2925
 
                srv_main_thread_op_info = "master purging";
2926
 
 
2927
 
                srv_master_do_purge();
2928
 
        }
 
2721
 
 
2722
                srv_main_thread_op_info = "purging";
 
2723
                n_pages_purged = trx_purge();
 
2724
 
 
2725
                /* Flush logs if needed */
 
2726
                srv_sync_log_buffer_in_background();
 
2727
 
 
2728
        } while (n_pages_purged);
2929
2729
 
2930
2730
        srv_main_thread_op_info = "reserving kernel mutex";
2931
2731
 
2943
2743
        } else {
2944
2744
                /* This should do an amount of IO similar to the number of
2945
2745
                dirty pages that will be flushed in the call to
2946
 
                buf_flush_list below. Otherwise, the system favors
 
2746
                buf_flush_batch below. Otherwise, the system favors
2947
2747
                clean pages over cleanup throughput. */
2948
2748
                n_bytes_merged = ibuf_contract_for_n_pages(FALSE,
2949
2749
                                                           PCT_IO(100));
2962
2762
        srv_main_thread_op_info = "flushing buffer pool pages";
2963
2763
        srv_main_flush_loops++;
2964
2764
        if (srv_fast_shutdown < 2) {
2965
 
                n_pages_flushed = buf_flush_list(
2966
 
                          PCT_IO(100), IB_ULONGLONG_MAX);
 
2765
                n_pages_flushed = buf_flush_batch(BUF_FLUSH_LIST,
 
2766
                                                  PCT_IO(100),
 
2767
                                                  IB_ULONGLONG_MAX);
2967
2768
        } else {
2968
2769
                /* In the fastest shutdown we do not flush the buffer pool
2969
2770
                to data files: we set n_pages_flushed to 0 artificially. */
2981
2782
        mutex_exit(&kernel_mutex);
2982
2783
 
2983
2784
        srv_main_thread_op_info = "waiting for buffer pool flush to end";
2984
 
        buf_flush_wait_batch_end(NULL, BUF_FLUSH_LIST);
 
2785
        buf_flush_wait_batch_end(BUF_FLUSH_LIST);
2985
2786
 
2986
2787
        /* Flush logs if needed */
2987
2788
        srv_sync_log_buffer_in_background();
3068
2869
                already when the event wait ends */
3069
2870
 
3070
2871
                os_thread_exit(NULL);
3071
 
 
3072
2872
        }
3073
2873
 
3074
2874
        /* When there is user activity, InnoDB will set the event and the
3077
2877
        goto loop;
3078
2878
 
3079
2879
 
3080
 
#if !defined(__SUNPRO_C)
 
2880
#if (!defined(__SUNPRO_C) && !defined(__SUNPRO_CC))
3081
2881
        OS_THREAD_DUMMY_RETURN; /* Not reached, avoid compiler warning */
3082
2882
#endif
3083
2883
}
3084
 
 
3085
 
/*********************************************************************//**
3086
 
Asynchronous purge thread.
3087
 
@return a dummy parameter */
3088
 
UNIV_INTERN
3089
 
os_thread_ret_t
3090
 
srv_purge_thread(
3091
 
/*=============*/
3092
 
        void*   /*arg __attribute__((unused))*/)        /*!< in: a dummy parameter
3093
 
                                                required by os_thread_create */
3094
 
{
3095
 
        srv_slot_t*     slot;
3096
 
        ulint           slot_no = ULINT_UNDEFINED;
3097
 
        ulint           n_total_purged = ULINT_UNDEFINED;
3098
 
 
3099
 
        ut_a(srv_n_purge_threads == 1);
3100
 
 
3101
 
#ifdef UNIV_DEBUG_THREAD_CREATION
3102
 
        fprintf(stderr, "InnoDB: Purge thread running, id %lu\n",
3103
 
                os_thread_pf(os_thread_get_curr_id()));
3104
 
#endif /* UNIV_DEBUG_THREAD_CREATION */
3105
 
 
3106
 
        mutex_enter(&kernel_mutex);
3107
 
 
3108
 
        slot_no = srv_table_reserve_slot(SRV_WORKER);
3109
 
 
3110
 
        slot = srv_table_get_nth_slot(slot_no);
3111
 
 
3112
 
        ++srv_n_threads_active[SRV_WORKER];
3113
 
 
3114
 
        mutex_exit(&kernel_mutex);
3115
 
 
3116
 
        while (srv_shutdown_state != SRV_SHUTDOWN_EXIT_THREADS) {
3117
 
 
3118
 
                ulint   n_pages_purged;
3119
 
 
3120
 
                /* If there are very few records to purge or the last
3121
 
                purge didn't purge any records then wait for activity.
3122
 
                We peek at the history len without holding any mutex
3123
 
                because in the worst case we will end up waiting for
3124
 
                the next purge event. */
3125
 
                if (trx_sys->rseg_history_len < srv_purge_batch_size
3126
 
                    || n_total_purged == 0) {
3127
 
 
3128
 
                        os_event_t      event;
3129
 
 
3130
 
                        mutex_enter(&kernel_mutex);
3131
 
 
3132
 
                        event = srv_suspend_thread();
3133
 
 
3134
 
                        mutex_exit(&kernel_mutex);
3135
 
 
3136
 
                        os_event_wait(event);
3137
 
                }
3138
 
 
3139
 
                /* Check for shutdown and whether we should do purge at all. */
3140
 
                if (srv_force_recovery >= SRV_FORCE_NO_BACKGROUND
3141
 
                    || srv_shutdown_state != 0
3142
 
                    || srv_fast_shutdown) {
3143
 
 
3144
 
                        break;
3145
 
                }
3146
 
 
3147
 
                n_total_purged = 0;
3148
 
 
3149
 
                /* Purge until there are no more records to purge and there is
3150
 
                no change in configuration or server state. */
3151
 
                do {
3152
 
                        n_pages_purged = trx_purge(srv_purge_batch_size);
3153
 
 
3154
 
                        n_total_purged += n_pages_purged;
3155
 
 
3156
 
                } while (n_pages_purged > 0 && !srv_fast_shutdown);
3157
 
 
3158
 
                srv_sync_log_buffer_in_background();
3159
 
        }
3160
 
 
3161
 
        mutex_enter(&kernel_mutex);
3162
 
 
3163
 
        ut_ad(srv_table_get_nth_slot(slot_no) == slot);
3164
 
 
3165
 
        /* Decrement the active count. */
3166
 
        srv_suspend_thread();
3167
 
 
3168
 
        slot->in_use = FALSE;
3169
 
 
3170
 
        /* Free the thread local memory. */
3171
 
        thr_local_free(os_thread_get_curr_id());
3172
 
 
3173
 
        mutex_exit(&kernel_mutex);
3174
 
 
3175
 
#ifdef UNIV_DEBUG_THREAD_CREATION
3176
 
        fprintf(stderr, "InnoDB: Purge thread exiting, id %lu\n",
3177
 
                os_thread_pf(os_thread_get_curr_id()));
3178
 
#endif /* UNIV_DEBUG_THREAD_CREATION */
3179
 
 
3180
 
        /* We count the number of threads in os_thread_exit(). A created
3181
 
        thread should always use that to exit and not use return() to exit. */
3182
 
        os_thread_exit(NULL);
3183
 
 
3184
 
        OS_THREAD_DUMMY_RETURN; /* Not reached, avoid compiler warning */
3185
 
}
3186
 
 
3187
 
/**********************************************************************//**
3188
 
Enqueues a task to server task queue and releases a worker thread, if there
3189
 
is a suspended one. */
3190
 
UNIV_INTERN
3191
 
void
3192
 
srv_que_task_enqueue_low(
3193
 
/*=====================*/
3194
 
        que_thr_t*      thr)    /*!< in: query thread */
3195
 
{
3196
 
        ut_ad(thr);
3197
 
 
3198
 
        mutex_enter(&kernel_mutex);
3199
 
 
3200
 
        UT_LIST_ADD_LAST(queue, srv_sys->tasks, thr);
3201
 
 
3202
 
        srv_release_threads(SRV_WORKER, 1);
3203
 
 
3204
 
        mutex_exit(&kernel_mutex);
3205
 
}