Commit 57b6cb39 authored by Vladislav Vaintroub's avatar Vladislav Vaintroub

Further review points and simplify Windows implementation

parent 7ed6530a
...@@ -934,7 +934,7 @@ lower-case-table-names 1 ...@@ -934,7 +934,7 @@ lower-case-table-names 1
master-info-file master.info master-info-file master.info
master-retry-count 86400 master-retry-count 86400
master-verify-checksum FALSE master-verify-checksum FALSE
max-allowed-packet 8388608 max-allowed-packet 1048576
max-binlog-cache-size 18446744073709547520 max-binlog-cache-size 18446744073709547520
max-binlog-size 1073741824 max-binlog-size 1073741824
max-binlog-stmt-cache-size 18446744073709547520 max-binlog-stmt-cache-size 18446744073709547520
...@@ -945,7 +945,7 @@ max-error-count 64 ...@@ -945,7 +945,7 @@ max-error-count 64
max-heap-table-size 16777216 max-heap-table-size 16777216
max-join-size 18446744073709551615 max-join-size 18446744073709551615
max-length-for-sort-data 1024 max-length-for-sort-data 1024
max-long-data-size 8388608 max-long-data-size 1048576
max-prepared-stmt-count 16382 max-prepared-stmt-count 16382
max-relay-log-size 0 max-relay-log-size 0
max-seeks-for-key 18446744073709551615 max-seeks-for-key 18446744073709551615
......
...@@ -3918,6 +3918,7 @@ extern "C" bool thd_sqlcom_can_generate_row_events(const MYSQL_THD thd) ...@@ -3918,6 +3918,7 @@ extern "C" bool thd_sqlcom_can_generate_row_events(const MYSQL_THD thd)
SYNOPSIS SYNOPSIS
thd_wait_begin() thd_wait_begin()
thd Thread object thd Thread object
Can be NULL, in this case current THD is used.
wait_type Type of wait wait_type Type of wait
1 -- short wait (e.g. for mutex) 1 -- short wait (e.g. for mutex)
2 -- medium wait (e.g. for disk io) 2 -- medium wait (e.g. for disk io)
...@@ -3945,6 +3946,7 @@ extern "C" void thd_wait_begin(MYSQL_THD thd, int wait_type) ...@@ -3945,6 +3946,7 @@ extern "C" void thd_wait_begin(MYSQL_THD thd, int wait_type)
when they waking up from a sleep/stall. when they waking up from a sleep/stall.
@param thd Thread handle @param thd Thread handle
Can be NULL, in this case current THD is used.
*/ */
extern "C" void thd_wait_end(MYSQL_THD thd) extern "C" void thd_wait_end(MYSQL_THD thd)
{ {
......
...@@ -2245,7 +2245,7 @@ static bool fix_threadpool_stall_limit(sys_var*, THD*, enum_var_type) ...@@ -2245,7 +2245,7 @@ static bool fix_threadpool_stall_limit(sys_var*, THD*, enum_var_type)
#ifdef _WIN32 #ifdef _WIN32
static Sys_var_uint Sys_threadpool_min_threads( static Sys_var_uint Sys_threadpool_min_threads(
"thread_pool_min_threads", "thread_pool_min_threads",
"Minimuim number of threads in the thread pool.", "Minimum number of threads in the thread pool.",
GLOBAL_VAR(threadpool_min_threads), CMD_LINE(REQUIRED_ARG), GLOBAL_VAR(threadpool_min_threads), CMD_LINE(REQUIRED_ARG),
VALID_RANGE(1, 256), DEFAULT(1), BLOCK_SIZE(1), VALID_RANGE(1, 256), DEFAULT(1), BLOCK_SIZE(1),
NO_MUTEX_GUARD, NOT_IN_BINLOG, ON_CHECK(0), NO_MUTEX_GUARD, NOT_IN_BINLOG, ON_CHECK(0),
...@@ -2267,8 +2267,9 @@ static Sys_var_uint Sys_threadpool_oversubscribe( ...@@ -2267,8 +2267,9 @@ static Sys_var_uint Sys_threadpool_oversubscribe(
); );
static Sys_var_uint Sys_threadpool_size( static Sys_var_uint Sys_threadpool_size(
"thread_pool_size", "thread_pool_size",
"Number of concurrently executing threads in the pool. " "Number of thread groups in the pool. "
"Leaving value default (0) sets it to the number of processors.", "This parameter is roughly equivalent to maximum number of concurrently "
"executing threads (threads in a waiting state do not count as executing).",
GLOBAL_VAR(threadpool_size), CMD_LINE(REQUIRED_ARG), GLOBAL_VAR(threadpool_size), CMD_LINE(REQUIRED_ARG),
VALID_RANGE(1, MAX_THREAD_GROUPS), DEFAULT(my_getncpus()), BLOCK_SIZE(1), VALID_RANGE(1, MAX_THREAD_GROUPS), DEFAULT(my_getncpus()), BLOCK_SIZE(1),
NO_MUTEX_GUARD, NOT_IN_BINLOG, ON_CHECK(0), NO_MUTEX_GUARD, NOT_IN_BINLOG, ON_CHECK(0),
......
This diff is collapsed.
...@@ -17,6 +17,30 @@ ...@@ -17,6 +17,30 @@
#include <windows.h> #include <windows.h>
/*
Threadpool API is not available on XP. We still want to compile a single
version on Windows, but use the latest functionality if available.
We cannot use threadpool functionality directly, since executable won't
start on XP and loader will complain about missing symbols.
We solve using the usual way it is done on Windows, i.e with dynamic loading.
We'll need to load a lot of function, and make this less painful with the
WEAK_SYMBOL macro below
*/
/*
WEAK_SYMBOL(return_type, function_name, argument_type1,..,argument_typeN)
Declare and load function pointer from kernel32. The name of the static
variable that holds the function pointer is my_<original function name>
This should be combined with
#define <original function name> my_<original function name>
so that one could use Widows APIs transparently, without worrying whether
they are present in a particular version or not.
Of course, prior to use of any function there should be a check for correct
Windows version, or check whether function pointer is not NULL.
*/
#define WEAK_SYMBOL(return_type, function, ...) \ #define WEAK_SYMBOL(return_type, function, ...) \
typedef return_type (WINAPI *pFN_##function)(__VA_ARGS__); \ typedef return_type (WINAPI *pFN_##function)(__VA_ARGS__); \
static pFN_##function my_##function = (pFN_##function) \ static pFN_##function my_##function = (pFN_##function) \
...@@ -110,9 +134,7 @@ WEAK_SYMBOL(VOID, CloseThreadpoolWork, PTP_WORK pwk); ...@@ -110,9 +134,7 @@ WEAK_SYMBOL(VOID, CloseThreadpoolWork, PTP_WORK pwk);
WEAK_SYMBOL(BOOL, SetThreadpoolStackInformation, PTP_POOL, WEAK_SYMBOL(BOOL, SetThreadpoolStackInformation, PTP_POOL,
PTP_POOL_STACK_INFORMATION); PTP_POOL_STACK_INFORMATION);
#define SetThreadpoolStackInformation my_SetThreadpoolStackInformation #define SetThreadpoolStackInformation my_SetThreadpoolStackInformation
#endif #else /* _MSC_VER < 1600 */
#if _MSC_VER < 1600
#define SetThreadpoolCallbackPriority(env,prio) #define SetThreadpoolCallbackPriority(env,prio)
typedef enum _TP_CALLBACK_PRIORITY { typedef enum _TP_CALLBACK_PRIORITY {
TP_CALLBACK_PRIORITY_HIGH, TP_CALLBACK_PRIORITY_HIGH,
...@@ -158,8 +180,6 @@ static void CALLBACK shm_read_callback(PTP_CALLBACK_INSTANCE instance, ...@@ -158,8 +180,6 @@ static void CALLBACK shm_read_callback(PTP_CALLBACK_INSTANCE instance,
static void CALLBACK shm_close_callback(PTP_CALLBACK_INSTANCE instance, static void CALLBACK shm_close_callback(PTP_CALLBACK_INSTANCE instance,
PVOID Context, PTP_WAIT wait,TP_WAIT_RESULT wait_result); PVOID Context, PTP_WAIT wait,TP_WAIT_RESULT wait_result);
#define CONNECTION_SIGNATURE 0xAFFEAFFE
static void check_thread_init(); static void check_thread_init();
/* Get current time as Windows time */ /* Get current time as Windows time */
...@@ -178,21 +198,19 @@ static ulonglong now() ...@@ -178,21 +198,19 @@ static ulonglong now()
struct connection_t struct connection_t
{ {
THD *thd; THD *thd;
bool logged_in;
HANDLE handle; HANDLE handle;
OVERLAPPED overlapped; OVERLAPPED overlapped;
/* absolute time for wait timeout (as Windows time) */ /* absolute time for wait timeout (as Windows time) */
volatile ulonglong timeout; volatile ulonglong timeout;
PTP_CLEANUP_GROUP cleanup_group; PTP_CLEANUP_GROUP cleanup_group;
TP_CALLBACK_ENVIRON callback_environ; TP_CALLBACK_ENVIRON callback_environ;
PTP_IO io; PTP_IO io;
PTP_TIMER timer; PTP_TIMER timer;
PTP_WAIT shm_read; PTP_WAIT shm_read;
bool logged_in;
}; };
void init_connection(connection_t *connection) void init_connection(connection_t *connection)
{ {
connection->logged_in = false; connection->logged_in = false;
...@@ -208,6 +226,7 @@ void init_connection(connection_t *connection) ...@@ -208,6 +226,7 @@ void init_connection(connection_t *connection)
connection->thd = 0; connection->thd = 0;
} }
int init_io(connection_t *connection, THD *thd) int init_io(connection_t *connection, THD *thd)
{ {
connection->thd= thd; connection->thd= thd;
...@@ -237,7 +256,7 @@ int init_io(connection_t *connection, THD *thd) ...@@ -237,7 +256,7 @@ int init_io(connection_t *connection, THD *thd)
if (connection->handle) if (connection->handle)
{ {
/* Performance tweaks (s. MSDN documentation)*/ /* Performance tweaks (s. MSDN documentation)*/
UCHAR flags = FILE_SKIP_SET_EVENT_ON_HANDLE; UCHAR flags= FILE_SKIP_SET_EVENT_ON_HANDLE;
if (skip_completion_port_on_success) if (skip_completion_port_on_success)
{ {
flags |= FILE_SKIP_COMPLETION_PORT_ON_SUCCESS; flags |= FILE_SKIP_COMPLETION_PORT_ON_SUCCESS;
...@@ -245,7 +264,7 @@ int init_io(connection_t *connection, THD *thd) ...@@ -245,7 +264,7 @@ int init_io(connection_t *connection, THD *thd)
(void)SetFileCompletionNotificationModes(connection->handle, flags); (void)SetFileCompletionNotificationModes(connection->handle, flags);
/* Assign io completion callback */ /* Assign io completion callback */
connection->io = CreateThreadpoolIo(connection->handle, connection->io= CreateThreadpoolIo(connection->handle,
io_completion_callback, connection, &connection->callback_environ); io_completion_callback, connection, &connection->callback_environ);
if(!connection->io) if(!connection->io)
{ {
...@@ -253,7 +272,7 @@ int init_io(connection_t *connection, THD *thd) ...@@ -253,7 +272,7 @@ int init_io(connection_t *connection, THD *thd)
return -1; return -1;
} }
} }
connection->timer = CreateThreadpoolTimer(timer_callback, connection, connection->timer= CreateThreadpoolTimer(timer_callback, connection,
&connection->callback_environ); &connection->callback_environ);
if (!connection->timer) if (!connection->timer)
{ {
...@@ -354,6 +373,7 @@ int start_io(connection_t *connection, PTP_CALLBACK_INSTANCE instance) ...@@ -354,6 +373,7 @@ int start_io(connection_t *connection, PTP_CALLBACK_INSTANCE instance)
return -1; return -1;
} }
int login(connection_t *connection, PTP_CALLBACK_INSTANCE instance) int login(connection_t *connection, PTP_CALLBACK_INSTANCE instance)
{ {
if (threadpool_add_connection(connection->thd) == 0 if (threadpool_add_connection(connection->thd) == 0
...@@ -380,21 +400,6 @@ void set_wait_timeout(connection_t *connection, ulonglong old_timeout) ...@@ -380,21 +400,6 @@ void set_wait_timeout(connection_t *connection, ulonglong old_timeout)
connection->timeout = new_timeout; connection->timeout = new_timeout;
} }
/*
Terminates (idle) connection by closing the socket.
This will activate io_completion_callback() in a different thread
*/
void post_kill_notification(connection_t *connection)
{
check_thread_init();
THD *thd=connection->thd;
mysql_mutex_lock(&thd->LOCK_thd_data);
thd->killed = KILL_CONNECTION;
vio_shutdown(thd->net.vio, SHUT_RDWR);
thd->mysys_var= NULL;
mysql_mutex_unlock(&thd->LOCK_thd_data);
}
/* Connection destructor */ /* Connection destructor */
void destroy_connection(connection_t *connection) void destroy_connection(connection_t *connection)
...@@ -438,7 +443,6 @@ static void check_thread_init() ...@@ -438,7 +443,6 @@ static void check_thread_init()
if (FlsGetValue(fls) == NULL) if (FlsGetValue(fls) == NULL)
{ {
FlsSetValue(fls, (void *)1); FlsSetValue(fls, (void *)1);
my_thread_init();
thread_created++; thread_created++;
InterlockedIncrement((volatile long *)&tp_stats.num_worker_threads); InterlockedIncrement((volatile long *)&tp_stats.num_worker_threads);
} }
...@@ -446,28 +450,14 @@ static void check_thread_init() ...@@ -446,28 +450,14 @@ static void check_thread_init()
/* /*
Take care of proper cleanup when threadpool threads exit. Decrement number of threads when a thread exits .
We do not control how threads are created, thus it is our responsibility to On Windows, FlsAlloc() provides the thread destruction callbacks.
check that my_thread_init() is called on thread initialization and
my_thread_end() on thread destruction. On Windows, FlsAlloc() provides the
thread destruction callbacks.
*/ */
static VOID WINAPI thread_destructor(void *data) static VOID WINAPI thread_destructor(void *data)
{ {
if(data) if(data)
{ {
if (InterlockedDecrement((volatile long *)&tp_stats.num_worker_threads) >= 0) InterlockedDecrement((volatile long *)&tp_stats.num_worker_threads);
{
/*
The above check for number of thread >= 0 is due to shutdown code (
see tp_end()) where we forcefully set num_worker_threads to 0, even
if not all threads have shut down yet to the point they would ran Fls
destructors, even after CloseThreadpool(). See also comment in tp_end().
*/
mysql_mutex_lock(&LOCK_thread_count);
my_thread_end();
mysql_mutex_unlock(&LOCK_thread_count);
}
} }
} }
...@@ -507,7 +497,7 @@ bool tp_init(void) ...@@ -507,7 +497,7 @@ bool tp_init(void)
{ {
TP_POOL_STACK_INFORMATION stackinfo; TP_POOL_STACK_INFORMATION stackinfo;
stackinfo.StackCommit = 0; stackinfo.StackCommit = 0;
stackinfo.StackReserve = my_thread_stack_size; stackinfo.StackReserve = (SIZE_T)my_thread_stack_size;
if (!SetThreadpoolStackInformation(pool, &stackinfo)) if (!SetThreadpoolStackInformation(pool, &stackinfo))
{ {
tp_log_warning("Can't set threadpool stack size", tp_log_warning("Can't set threadpool stack size",
...@@ -520,45 +510,19 @@ bool tp_init(void) ...@@ -520,45 +510,19 @@ bool tp_init(void)
} }
/* /**
Scheduler callback : Destroy the scheduler. Scheduler callback : Destroy the scheduler.
*/ */
extern "C" uint THR_thread_count;
extern "C" mysql_mutex_t THR_LOCK_threads;
extern "C" mysql_cond_t THR_COND_threads;
void tp_end(void) void tp_end(void)
{ {
if(pool) if(pool)
{ {
SetThreadpoolThreadMaximum(pool, 0); SetThreadpoolThreadMaximum(pool, 0);
CloseThreadpool(pool); CloseThreadpool(pool);
/*
Tell my_global_thread_end() we're complete.
This would not be necessary if CloseThreadpool() would synchronously
release all threads and wait until they disappear and call all their FLS
destructors . However, threads in the pool are released asynchronously
and might spend some time in the CRT shutdown code. Thus zero
num_worker_threads, to avoid thread destructor's my_thread_end()s after
this point.
*/
LONG remaining_threads=
InterlockedExchange( (volatile long *)&tp_stats.num_worker_threads, 0);
if (remaining_threads)
{
mysql_mutex_lock(&THR_LOCK_threads);
THR_thread_count -= remaining_threads;
mysql_cond_signal(&THR_COND_threads);
mysql_mutex_unlock(&THR_LOCK_threads);
}
} }
} }
/* /**
Notify pool about connection being killed. Notify pool about connection being killed.
*/ */
void tp_post_kill_notification(THD *thd) void tp_post_kill_notification(THD *thd)
...@@ -606,7 +570,7 @@ error: ...@@ -606,7 +570,7 @@ error:
DisassociateCurrentThreadFromCallback(instance); DisassociateCurrentThreadFromCallback(instance);
destroy_connection(connection); destroy_connection(connection);
my_free(connection); free(connection);
} }
...@@ -623,7 +587,7 @@ static void CALLBACK login_callback(PTP_CALLBACK_INSTANCE instance, ...@@ -623,7 +587,7 @@ static void CALLBACK login_callback(PTP_CALLBACK_INSTANCE instance,
if (login(connection, instance) != 0) if (login(connection, instance) != 0)
{ {
destroy_connection(connection); destroy_connection(connection);
my_free(connection); free(connection);
} }
} }
...@@ -688,9 +652,8 @@ static void CALLBACK shm_read_callback(PTP_CALLBACK_INSTANCE instance, ...@@ -688,9 +652,8 @@ static void CALLBACK shm_read_callback(PTP_CALLBACK_INSTANCE instance,
*/ */
void tp_add_connection(THD *thd) void tp_add_connection(THD *thd)
{ {
bool success = false; connection_t *con = (connection_t *)malloc(sizeof(connection_t));
connection_t *con = (connection_t *)my_malloc(sizeof(connection_t), 0);
if (con) if (con)
threads.append(thd); threads.append(thd);
mysql_mutex_unlock(&LOCK_thread_count); mysql_mutex_unlock(&LOCK_thread_count);
...@@ -698,6 +661,7 @@ void tp_add_connection(THD *thd) ...@@ -698,6 +661,7 @@ void tp_add_connection(THD *thd)
if(!con) if(!con)
{ {
tp_log_warning("Allocation failed", "tp_add_connection"); tp_log_warning("Allocation failed", "tp_add_connection");
threadpool_remove_connection(thd);
return; return;
} }
...@@ -718,8 +682,7 @@ void tp_add_connection(THD *thd) ...@@ -718,8 +682,7 @@ void tp_add_connection(THD *thd)
} }
/**
/*
Sets the number of idle threads the thread pool maintains in anticipation of new Sets the number of idle threads the thread pool maintains in anticipation of new
requests. requests.
*/ */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment