2010-04-14 09:30:57 +02:00
|
|
|
/* Copyright (C) 2004, 2010 Oracle and/or its affiliates. All rights reserved.
|
2006-05-22 20:46:13 +02:00
|
|
|
|
|
|
|
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
|
2006-12-27 02:23:51 +01:00
|
|
|
the Free Software Foundation; version 2 of the License.
|
2006-05-22 20:46:13 +02:00
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
2010-04-14 09:30:57 +02:00
|
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
2006-05-22 20:46:13 +02:00
|
|
|
|
2010-03-31 16:05:33 +02:00
|
|
|
#include "sql_priv.h"
|
|
|
|
#include "unireg.h"
|
|
|
|
#include "event_scheduler.h"
|
2006-07-12 10:37:30 +02:00
|
|
|
#include "events.h"
|
|
|
|
#include "event_data_objects.h"
|
|
|
|
#include "event_queue.h"
|
2007-01-29 18:46:29 +01:00
|
|
|
#include "event_db_repository.h"
|
2010-03-31 16:05:33 +02:00
|
|
|
#include "sql_connect.h" // init_new_connection_handler_thread
|
|
|
|
#include "sql_acl.h" // SUPER_ACL
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2007-08-15 17:08:44 +02:00
|
|
|
/**
|
|
|
|
@addtogroup Event_Scheduler
|
|
|
|
@{
|
|
|
|
*/
|
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
#ifdef __GNUC__
|
|
|
|
#if __GNUC__ >= 2
|
|
|
|
#define SCHED_FUNC __FUNCTION__
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define SCHED_FUNC "<unknown>"
|
|
|
|
#endif
|
|
|
|
|
2006-07-17 16:52:45 +02:00
|
|
|
#define LOCK_DATA() lock_data(SCHED_FUNC, __LINE__)
|
|
|
|
#define UNLOCK_DATA() unlock_data(SCHED_FUNC, __LINE__)
|
|
|
|
#define COND_STATE_WAIT(mythd, abstime, msg) \
|
|
|
|
cond_wait(mythd, abstime, msg, SCHED_FUNC, __LINE__)
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
extern pthread_attr_t connection_attrib;
|
|
|
|
|
2007-01-29 18:46:29 +01:00
|
|
|
|
|
|
|
Event_db_repository *Event_worker_thread::db_repository;
|
|
|
|
|
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
static
|
2006-08-17 17:44:16 +02:00
|
|
|
const LEX_STRING scheduler_states_names[] =
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2006-08-17 17:44:16 +02:00
|
|
|
{ C_STRING_WITH_LEN("INITIALIZED") },
|
|
|
|
{ C_STRING_WITH_LEN("RUNNING") },
|
|
|
|
{ C_STRING_WITH_LEN("STOPPING") }
|
2006-07-12 10:37:30 +02:00
|
|
|
};
|
|
|
|
|
2006-07-13 16:24:55 +02:00
|
|
|
struct scheduler_param {
|
|
|
|
THD *thd;
|
|
|
|
Event_scheduler *scheduler;
|
|
|
|
};
|
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Prints the stack of infos, warnings, errors from thd to
|
|
|
|
the console so it can be fetched by the logs-into-tables and
|
|
|
|
checked later.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
evex_print_warnings
|
|
|
|
thd Thread used during the execution of the event
|
|
|
|
et The event itself
|
|
|
|
*/
|
|
|
|
|
2007-01-29 18:46:29 +01:00
|
|
|
void
|
|
|
|
Event_worker_thread::print_warnings(THD *thd, Event_job_data *et)
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
|
|
|
MYSQL_ERROR *err;
|
|
|
|
DBUG_ENTER("evex_print_warnings");
|
2009-09-10 11:18:29 +02:00
|
|
|
if (thd->warning_info->is_empty())
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
|
|
|
|
char msg_buf[10 * STRING_BUFFER_USUAL_SIZE];
|
|
|
|
char prefix_buf[5 * STRING_BUFFER_USUAL_SIZE];
|
|
|
|
String prefix(prefix_buf, sizeof(prefix_buf), system_charset_info);
|
|
|
|
prefix.length(0);
|
2007-03-23 17:14:08 +01:00
|
|
|
prefix.append("Event Scheduler: [");
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2007-04-05 13:24:34 +02:00
|
|
|
prefix.append(et->definer.str, et->definer.length, system_charset_info);
|
2006-07-12 10:37:30 +02:00
|
|
|
prefix.append("][", 2);
|
2007-04-05 13:24:34 +02:00
|
|
|
prefix.append(et->dbname.str, et->dbname.length, system_charset_info);
|
2006-07-12 10:37:30 +02:00
|
|
|
prefix.append('.');
|
2007-04-05 13:24:34 +02:00
|
|
|
prefix.append(et->name.str, et->name.length, system_charset_info);
|
2006-07-12 10:37:30 +02:00
|
|
|
prefix.append("] ", 2);
|
|
|
|
|
2009-09-10 11:18:29 +02:00
|
|
|
List_iterator_fast<MYSQL_ERROR> it(thd->warning_info->warn_list());
|
2006-07-12 10:37:30 +02:00
|
|
|
while ((err= it++))
|
|
|
|
{
|
|
|
|
String err_msg(msg_buf, sizeof(msg_buf), system_charset_info);
|
|
|
|
/* set it to 0 or we start adding at the end. That's the trick ;) */
|
|
|
|
err_msg.length(0);
|
|
|
|
err_msg.append(prefix);
|
2009-09-10 11:18:29 +02:00
|
|
|
err_msg.append(err->get_message_text(),
|
|
|
|
err->get_message_octet_length(), system_charset_info);
|
|
|
|
DBUG_ASSERT(err->get_level() < 3);
|
|
|
|
(sql_print_message_handlers[err->get_level()])("%*s", err_msg.length(),
|
|
|
|
err_msg.c_ptr());
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
/*
|
|
|
|
Performs post initialization of structures in a new thread.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
post_init_event_thread()
|
|
|
|
thd Thread
|
2007-03-01 21:47:28 +01:00
|
|
|
|
|
|
|
NOTES
|
|
|
|
Before this is called, one should not do any DBUG_XXX() calls.
|
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
bool
|
|
|
|
post_init_event_thread(THD *thd)
|
|
|
|
{
|
2007-03-01 21:47:28 +01:00
|
|
|
(void) init_new_connection_handler_thread();
|
2006-07-13 10:59:58 +02:00
|
|
|
if (init_thr_lock() || thd->store_globals())
|
|
|
|
{
|
|
|
|
thd->cleanup();
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_lock(&LOCK_thread_count);
|
2006-08-11 16:51:50 +02:00
|
|
|
threads.append(thd);
|
|
|
|
thread_count++;
|
2009-10-12 11:00:39 +02:00
|
|
|
inc_thread_running();
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_unlock(&LOCK_thread_count);
|
2006-07-13 10:59:58 +02:00
|
|
|
return FALSE;
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Cleans up the THD and the threaded environment of the thread.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
deinit_event_thread()
|
|
|
|
thd Thread
|
|
|
|
*/
|
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
void
|
2006-07-12 10:37:30 +02:00
|
|
|
deinit_event_thread(THD *thd)
|
|
|
|
{
|
|
|
|
thd->proc_info= "Clearing";
|
|
|
|
DBUG_ASSERT(thd->net.buff != 0);
|
|
|
|
net_end(&thd->net);
|
2006-07-17 16:52:45 +02:00
|
|
|
DBUG_PRINT("exit", ("Event thread finishing"));
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_lock(&LOCK_thread_count);
|
2006-07-12 10:37:30 +02:00
|
|
|
thread_count--;
|
2009-10-12 11:00:39 +02:00
|
|
|
dec_thread_running();
|
2009-11-20 16:43:21 +01:00
|
|
|
delete thd;
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_cond_broadcast(&COND_thread_count);
|
|
|
|
mysql_mutex_unlock(&LOCK_thread_count);
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-13 16:24:55 +02:00
|
|
|
/*
|
2010-01-07 06:42:07 +01:00
|
|
|
Performs pre- mysql_thread_create() initialisation of THD. Do this
|
2006-07-13 16:24:55 +02:00
|
|
|
in the thread that will pass THD to the child thread. In the
|
|
|
|
child thread call post_init_event_thread().
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
pre_init_event_thread()
|
|
|
|
thd The THD of the thread. Has to be allocated by the caller.
|
|
|
|
|
|
|
|
NOTES
|
|
|
|
1. The host of the thead is my_localhost
|
|
|
|
2. thd->net is initted with NULL - no communication.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
pre_init_event_thread(THD* thd)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("pre_init_event_thread");
|
|
|
|
thd->client_capabilities= 0;
|
|
|
|
thd->security_ctx->master_access= 0;
|
|
|
|
thd->security_ctx->db_access= 0;
|
|
|
|
thd->security_ctx->host_or_ip= (char*)my_localhost;
|
|
|
|
my_net_init(&thd->net, NULL);
|
|
|
|
thd->security_ctx->set_user((char*)"event_scheduler");
|
|
|
|
thd->net.read_timeout= slave_net_timeout;
|
|
|
|
thd->slave_thread= 0;
|
2009-12-22 10:35:56 +01:00
|
|
|
thd->variables.option_bits|= OPTION_AUTO_IS_NULL;
|
2006-07-13 16:24:55 +02:00
|
|
|
thd->client_capabilities|= CLIENT_MULTI_RESULTS;
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_lock(&LOCK_thread_count);
|
2007-03-01 21:47:28 +01:00
|
|
|
thd->thread_id= thd->variables.pseudo_thread_id= thread_id++;
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_unlock(&LOCK_thread_count);
|
2006-07-13 16:24:55 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Guarantees that we will see the thread in SHOW PROCESSLIST though its
|
|
|
|
vio is NULL.
|
|
|
|
*/
|
|
|
|
|
|
|
|
thd->proc_info= "Initialized";
|
|
|
|
thd->set_time();
|
|
|
|
|
2010-02-24 18:04:00 +01:00
|
|
|
/* Do not use user-supplied timeout value for system threads. */
|
|
|
|
thd->variables.lock_wait_timeout= LONG_TIMEOUT;
|
|
|
|
|
2006-07-13 16:24:55 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
/*
|
|
|
|
Function that executes the scheduler,
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
event_scheduler_thread()
|
|
|
|
arg Pointer to `struct scheduler_param`
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
0 OK
|
|
|
|
*/
|
|
|
|
|
|
|
|
pthread_handler_t
|
|
|
|
event_scheduler_thread(void *arg)
|
|
|
|
{
|
|
|
|
/* needs to be first for thread_stack */
|
2007-03-01 21:47:28 +01:00
|
|
|
THD *thd= (THD *) ((struct scheduler_param *) arg)->thd;
|
2006-07-13 10:59:58 +02:00
|
|
|
Event_scheduler *scheduler= ((struct scheduler_param *) arg)->scheduler;
|
2007-03-01 21:47:28 +01:00
|
|
|
bool res;
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
thd->thread_stack= (char *)&thd; // remember where our stack is
|
2010-01-07 06:42:07 +01:00
|
|
|
|
|
|
|
mysql_thread_set_psi_id(thd->thread_id);
|
|
|
|
|
2007-03-01 21:47:28 +01:00
|
|
|
res= post_init_event_thread(thd);
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
DBUG_ENTER("event_scheduler_thread");
|
Bug#34043: Server loops excessively in _checkchunk() when safemalloc is enabled
Essentially, the problem is that safemalloc is excruciatingly
slow as it checks all allocated blocks for overrun at each
memory management primitive, yielding a almost exponential
slowdown for the memory management functions (malloc, realloc,
free). The overrun check basically consists of verifying some
bytes of a block for certain magic keys, which catches some
simple forms of overrun. Another minor problem is violation
of aliasing rules and that its own internal list of blocks
is prone to corruption.
Another issue with safemalloc is rather the maintenance cost
as the tool has a significant impact on the server code.
Given the magnitude of memory debuggers available nowadays,
especially those that are provided with the platform malloc
implementation, maintenance of a in-house and largely obsolete
memory debugger becomes a burden that is not worth the effort
due to its slowness and lack of support for detecting more
common forms of heap corruption.
Since there are third-party tools that can provide the same
functionality at a lower or comparable performance cost, the
solution is to simply remove safemalloc. Third-party tools
can provide the same functionality at a lower or comparable
performance cost.
The removal of safemalloc also allows a simplification of the
malloc wrappers, removing quite a bit of kludge: redefinition
of my_malloc, my_free and the removal of the unused second
argument of my_free. Since free() always check whether the
supplied pointer is null, redudant checks are also removed.
Also, this patch adds unit testing for my_malloc and moves
my_realloc implementation into the same file as the other
memory allocation primitives.
2010-07-08 23:20:08 +02:00
|
|
|
my_free(arg);
|
2007-03-01 21:47:28 +01:00
|
|
|
if (!res)
|
2006-07-13 10:59:58 +02:00
|
|
|
scheduler->run(thd);
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2010-02-02 13:17:21 +01:00
|
|
|
DBUG_LEAVE; // Against gcc warnings
|
2007-04-06 17:44:14 +02:00
|
|
|
my_thread_end();
|
2010-02-02 13:17:21 +01:00
|
|
|
return 0;
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-04-05 13:24:34 +02:00
|
|
|
/**
|
2007-03-23 16:14:03 +01:00
|
|
|
Function that executes an event in a child thread. Setups the
|
2006-07-12 10:37:30 +02:00
|
|
|
environment for the event execution and cleans after that.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
event_worker_thread()
|
|
|
|
arg The Event_job_data object to be processed
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
0 OK
|
|
|
|
*/
|
|
|
|
|
|
|
|
pthread_handler_t
|
|
|
|
event_worker_thread(void *arg)
|
|
|
|
{
|
2007-03-23 16:14:03 +01:00
|
|
|
THD *thd;
|
2007-01-29 18:46:29 +01:00
|
|
|
Event_queue_element_for_exec *event= (Event_queue_element_for_exec *)arg;
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
thd= event->thd;
|
|
|
|
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_thread_set_psi_id(thd->thread_id);
|
|
|
|
|
2007-01-29 18:46:29 +01:00
|
|
|
Event_worker_thread worker_thread;
|
2007-03-01 21:47:28 +01:00
|
|
|
worker_thread.run(thd, event);
|
2007-01-29 18:46:29 +01:00
|
|
|
|
2007-04-06 17:44:14 +02:00
|
|
|
my_thread_end();
|
2007-01-29 18:46:29 +01:00
|
|
|
return 0; // Can't return anything here
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-04-05 13:24:34 +02:00
|
|
|
/**
|
2007-03-23 16:14:03 +01:00
|
|
|
Function that executes an event in a child thread. Setups the
|
2007-01-29 18:46:29 +01:00
|
|
|
environment for the event execution and cleans after that.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_worker_thread::run()
|
|
|
|
thd Thread context
|
|
|
|
event The Event_queue_element_for_exec object to be processed
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
Event_worker_thread::run(THD *thd, Event_queue_element_for_exec *event)
|
|
|
|
{
|
2007-03-01 21:47:28 +01:00
|
|
|
/* needs to be first for thread_stack */
|
|
|
|
char my_stack;
|
2007-04-13 22:35:56 +02:00
|
|
|
Event_job_data job_data;
|
2007-03-01 21:47:28 +01:00
|
|
|
bool res;
|
|
|
|
|
|
|
|
thd->thread_stack= &my_stack; // remember where our stack is
|
|
|
|
res= post_init_event_thread(thd);
|
|
|
|
|
2007-01-29 18:46:29 +01:00
|
|
|
DBUG_ENTER("Event_worker_thread::run");
|
2007-07-30 10:33:50 +02:00
|
|
|
DBUG_PRINT("info", ("Time is %ld, THD: 0x%lx", (long) my_time(0), (long) thd));
|
2007-03-01 21:47:28 +01:00
|
|
|
|
|
|
|
if (res)
|
2007-01-29 18:46:29 +01:00
|
|
|
goto end;
|
|
|
|
|
2007-04-13 22:35:56 +02:00
|
|
|
if ((res= db_repository->load_named_event(thd, event->dbname, event->name,
|
|
|
|
&job_data)))
|
2007-01-29 18:46:29 +01:00
|
|
|
{
|
2007-04-13 22:35:56 +02:00
|
|
|
DBUG_PRINT("error", ("Got error from load_named_event"));
|
2007-01-29 18:46:29 +01:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
thd->enable_slow_log= TRUE;
|
|
|
|
|
2007-04-13 22:35:56 +02:00
|
|
|
res= job_data.execute(thd, event->dropped);
|
2007-01-29 18:46:29 +01:00
|
|
|
|
2007-04-13 22:35:56 +02:00
|
|
|
print_warnings(thd, &job_data);
|
2007-01-29 18:46:29 +01:00
|
|
|
|
2007-04-13 22:35:56 +02:00
|
|
|
if (res)
|
|
|
|
sql_print_information("Event Scheduler: "
|
|
|
|
"[%s].[%s.%s] event execution failed.",
|
|
|
|
job_data.definer.str,
|
|
|
|
job_data.dbname.str, job_data.name.str);
|
2007-01-29 18:46:29 +01:00
|
|
|
end:
|
2007-03-02 00:39:00 +01:00
|
|
|
DBUG_PRINT("info", ("Done with Event %s.%s", event->dbname.str,
|
2006-07-12 10:37:30 +02:00
|
|
|
event->name.str));
|
|
|
|
|
2007-01-29 18:46:29 +01:00
|
|
|
delete event;
|
2007-03-01 21:47:28 +01:00
|
|
|
deinit_event_thread(thd);
|
2007-06-10 11:16:02 +02:00
|
|
|
|
|
|
|
DBUG_VOID_RETURN;
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-04-05 13:24:34 +02:00
|
|
|
Event_scheduler::Event_scheduler(Event_queue *queue_arg)
|
2007-05-21 10:51:11 +02:00
|
|
|
:state(INITIALIZED),
|
2007-04-05 13:24:34 +02:00
|
|
|
scheduler_thd(NULL),
|
|
|
|
queue(queue_arg),
|
2007-05-21 10:51:11 +02:00
|
|
|
mutex_last_locked_at_line(0),
|
|
|
|
mutex_last_unlocked_at_line(0),
|
|
|
|
mutex_last_locked_in_func("n/a"),
|
|
|
|
mutex_last_unlocked_in_func("n/a"),
|
|
|
|
mutex_scheduler_data_locked(FALSE),
|
|
|
|
waiting_on_cond(FALSE),
|
2007-04-05 13:24:34 +02:00
|
|
|
started_events(0)
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_mutex_init(key_event_scheduler_LOCK_scheduler_state,
|
|
|
|
&LOCK_scheduler_state, MY_MUTEX_INIT_FAST);
|
|
|
|
mysql_cond_init(key_event_scheduler_COND_state, &COND_state, NULL);
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-04-05 13:24:34 +02:00
|
|
|
Event_scheduler::~Event_scheduler()
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2007-04-05 13:24:34 +02:00
|
|
|
stop(); /* does nothing if not running */
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_mutex_destroy(&LOCK_scheduler_state);
|
|
|
|
mysql_cond_destroy(&COND_state);
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Starts the scheduler (again). Creates a new THD and passes it to
|
|
|
|
a forked thread. Does not wait for acknowledgement from the new
|
|
|
|
thread that it has started. Asynchronous starting. Most of the
|
|
|
|
needed initializations are done in the current thread to minimize
|
|
|
|
the chance of failure in the spawned thread.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::start()
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
FALSE OK
|
|
|
|
TRUE Error (not reported)
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool
|
|
|
|
Event_scheduler::start()
|
|
|
|
{
|
|
|
|
THD *new_thd= NULL;
|
|
|
|
bool ret= FALSE;
|
|
|
|
pthread_t th;
|
2006-07-13 10:59:58 +02:00
|
|
|
struct scheduler_param *scheduler_param_value;
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_ENTER("Event_scheduler::start");
|
|
|
|
|
2006-07-13 16:24:55 +02:00
|
|
|
LOCK_DATA();
|
2006-11-27 00:47:38 +01:00
|
|
|
DBUG_PRINT("info", ("state before action %s", scheduler_states_names[state].str));
|
2006-07-12 10:37:30 +02:00
|
|
|
if (state > INITIALIZED)
|
|
|
|
goto end;
|
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
if (!(new_thd= new THD))
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2007-03-23 17:14:08 +01:00
|
|
|
sql_print_error("Event Scheduler: Cannot initialize the scheduler thread");
|
2006-07-12 10:37:30 +02:00
|
|
|
ret= TRUE;
|
|
|
|
goto end;
|
|
|
|
}
|
2006-07-13 10:59:58 +02:00
|
|
|
pre_init_event_thread(new_thd);
|
2006-07-12 10:37:30 +02:00
|
|
|
new_thd->system_thread= SYSTEM_THREAD_EVENT_SCHEDULER;
|
|
|
|
new_thd->command= COM_DAEMON;
|
|
|
|
|
2007-10-19 17:57:08 +02:00
|
|
|
/*
|
|
|
|
We should run the event scheduler thread under the super-user privileges.
|
|
|
|
In particular, this is needed to be able to lock the mysql.event table
|
|
|
|
for writing when the server is running in the read-only mode.
|
|
|
|
*/
|
|
|
|
new_thd->security_ctx->master_access |= SUPER_ACL;
|
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
scheduler_param_value=
|
|
|
|
(struct scheduler_param *)my_malloc(sizeof(struct scheduler_param), MYF(0));
|
|
|
|
scheduler_param_value->thd= new_thd;
|
|
|
|
scheduler_param_value->scheduler= this;
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2006-08-31 17:18:39 +02:00
|
|
|
scheduler_thd= new_thd;
|
|
|
|
DBUG_PRINT("info", ("Setting state go RUNNING"));
|
|
|
|
state= RUNNING;
|
2007-03-23 16:14:03 +01:00
|
|
|
DBUG_PRINT("info", ("Forking new thread for scheduler. THD: 0x%lx", (long) new_thd));
|
2010-01-07 06:42:07 +01:00
|
|
|
if (mysql_thread_create(key_thread_event_scheduler,
|
|
|
|
&th, &connection_attrib, event_scheduler_thread,
|
|
|
|
(void*)scheduler_param_value))
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("cannot create a new thread"));
|
|
|
|
state= INITIALIZED;
|
2006-08-31 17:18:39 +02:00
|
|
|
scheduler_thd= NULL;
|
2006-07-12 10:37:30 +02:00
|
|
|
ret= TRUE;
|
|
|
|
|
|
|
|
new_thd->proc_info= "Clearing";
|
|
|
|
DBUG_ASSERT(new_thd->net.buff != 0);
|
|
|
|
net_end(&new_thd->net);
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_lock(&LOCK_thread_count);
|
2006-07-12 10:37:30 +02:00
|
|
|
thread_count--;
|
2009-10-12 11:00:39 +02:00
|
|
|
dec_thread_running();
|
2009-11-20 16:43:21 +01:00
|
|
|
delete new_thd;
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_cond_broadcast(&COND_thread_count);
|
|
|
|
mysql_mutex_unlock(&LOCK_thread_count);
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
2006-08-31 17:18:39 +02:00
|
|
|
end:
|
|
|
|
UNLOCK_DATA();
|
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_RETURN(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
The main loop of the scheduler.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::run()
|
|
|
|
thd Thread
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
FALSE OK
|
|
|
|
TRUE Error (Serious error)
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool
|
|
|
|
Event_scheduler::run(THD *thd)
|
|
|
|
{
|
2006-08-17 14:22:59 +02:00
|
|
|
int res= FALSE;
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_ENTER("Event_scheduler::run");
|
|
|
|
|
2007-03-23 17:14:08 +01:00
|
|
|
sql_print_information("Event Scheduler: scheduler thread started with id %lu",
|
2006-08-31 17:18:39 +02:00
|
|
|
thd->thread_id);
|
2006-07-12 10:37:30 +02:00
|
|
|
/*
|
|
|
|
Recalculate the values in the queue because there could have been stops
|
|
|
|
in executions of the scheduler and some times could have passed by.
|
|
|
|
*/
|
|
|
|
queue->recalculate_activation_times(thd);
|
2006-08-31 17:18:39 +02:00
|
|
|
|
|
|
|
while (is_running())
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2007-01-29 18:46:29 +01:00
|
|
|
Event_queue_element_for_exec *event_name;
|
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
/* Gets a minimized version */
|
2007-01-29 18:46:29 +01:00
|
|
|
if (queue->get_top_for_execution_if_time(thd, &event_name))
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2007-03-23 17:14:08 +01:00
|
|
|
sql_print_information("Event Scheduler: "
|
|
|
|
"Serious error during getting next "
|
2006-08-31 17:18:39 +02:00
|
|
|
"event to execute. Stopping");
|
2006-07-12 10:37:30 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-03-02 00:39:00 +01:00
|
|
|
DBUG_PRINT("info", ("get_top_for_execution_if_time returned "
|
|
|
|
"event_name=0x%lx", (long) event_name));
|
2007-01-29 18:46:29 +01:00
|
|
|
if (event_name)
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2007-03-23 16:14:03 +01:00
|
|
|
if ((res= execute_top(event_name)))
|
2006-08-31 17:18:39 +02:00
|
|
|
break;
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-08-31 17:18:39 +02:00
|
|
|
DBUG_ASSERT(thd->killed);
|
|
|
|
DBUG_PRINT("info", ("job_data is NULL, the thread was killed"));
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
DBUG_PRINT("info", ("state=%s", scheduler_states_names[state].str));
|
|
|
|
}
|
2007-04-06 17:44:14 +02:00
|
|
|
|
2006-08-31 17:18:39 +02:00
|
|
|
LOCK_DATA();
|
2007-04-06 17:44:14 +02:00
|
|
|
deinit_event_thread(thd);
|
|
|
|
scheduler_thd= NULL;
|
2006-07-12 10:37:30 +02:00
|
|
|
state= INITIALIZED;
|
2010-04-14 09:30:57 +02:00
|
|
|
DBUG_PRINT("info", ("Broadcasting COND_state back to the stoppers"));
|
|
|
|
mysql_cond_broadcast(&COND_state);
|
2006-07-13 16:24:55 +02:00
|
|
|
UNLOCK_DATA();
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Creates a new THD instance and then forks a new thread, while passing
|
|
|
|
the THD pointer and job_data to it.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::execute_top()
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
FALSE OK
|
|
|
|
TRUE Error (Serious error)
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool
|
2007-03-23 16:14:03 +01:00
|
|
|
Event_scheduler::execute_top(Event_queue_element_for_exec *event_name)
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
|
|
|
THD *new_thd;
|
|
|
|
pthread_t th;
|
|
|
|
int res= 0;
|
|
|
|
DBUG_ENTER("Event_scheduler::execute_top");
|
2006-08-17 14:22:59 +02:00
|
|
|
if (!(new_thd= new THD()))
|
2006-07-12 10:37:30 +02:00
|
|
|
goto error;
|
|
|
|
|
2006-07-13 10:59:58 +02:00
|
|
|
pre_init_event_thread(new_thd);
|
2006-07-12 10:37:30 +02:00
|
|
|
new_thd->system_thread= SYSTEM_THREAD_EVENT_WORKER;
|
2007-01-29 18:46:29 +01:00
|
|
|
event_name->thd= new_thd;
|
2007-03-02 00:39:00 +01:00
|
|
|
DBUG_PRINT("info", ("Event %s@%s ready for start",
|
2007-01-29 18:46:29 +01:00
|
|
|
event_name->dbname.str, event_name->name.str));
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2007-03-16 15:31:07 +01:00
|
|
|
/*
|
|
|
|
TODO: should use thread pool here, preferably with an upper limit
|
|
|
|
on number of threads: if too many events are scheduled for the
|
|
|
|
same time, starting all of them at once won't help them run truly
|
|
|
|
in parallel (because of the great amount of synchronization), so
|
|
|
|
we may as well execute them in sequence, keeping concurrency at a
|
|
|
|
reasonable level.
|
|
|
|
*/
|
2006-07-12 10:37:30 +02:00
|
|
|
/* Major failure */
|
2010-01-07 06:42:07 +01:00
|
|
|
if ((res= mysql_thread_create(key_thread_event_worker,
|
|
|
|
&th, &connection_attrib, event_worker_thread,
|
|
|
|
event_name)))
|
2006-07-12 10:37:30 +02:00
|
|
|
goto error;
|
|
|
|
|
2006-08-31 17:18:39 +02:00
|
|
|
++started_events;
|
|
|
|
|
2007-03-02 00:39:00 +01:00
|
|
|
DBUG_PRINT("info", ("Event is in THD: 0x%lx", (long) new_thd));
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
|
|
|
|
error:
|
2007-03-02 00:39:00 +01:00
|
|
|
DBUG_PRINT("error", ("Event_scheduler::execute_top() res: %d", res));
|
2006-07-12 10:37:30 +02:00
|
|
|
if (new_thd)
|
|
|
|
{
|
|
|
|
new_thd->proc_info= "Clearing";
|
|
|
|
DBUG_ASSERT(new_thd->net.buff != 0);
|
|
|
|
net_end(&new_thd->net);
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_lock(&LOCK_thread_count);
|
2006-07-12 10:37:30 +02:00
|
|
|
thread_count--;
|
2009-10-12 11:00:39 +02:00
|
|
|
dec_thread_running();
|
2009-11-20 16:43:21 +01:00
|
|
|
delete new_thd;
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_cond_broadcast(&COND_thread_count);
|
|
|
|
mysql_mutex_unlock(&LOCK_thread_count);
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
2007-01-29 18:46:29 +01:00
|
|
|
delete event_name;
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-31 17:18:39 +02:00
|
|
|
/*
|
2006-09-01 14:08:50 +02:00
|
|
|
Checks whether the state of the scheduler is RUNNING
|
2006-08-31 17:18:39 +02:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::is_running()
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
TRUE RUNNING
|
|
|
|
FALSE Not RUNNING
|
|
|
|
*/
|
|
|
|
|
2006-09-01 14:15:47 +02:00
|
|
|
bool
|
2006-08-31 17:18:39 +02:00
|
|
|
Event_scheduler::is_running()
|
|
|
|
{
|
|
|
|
LOCK_DATA();
|
|
|
|
bool ret= (state == RUNNING);
|
|
|
|
UNLOCK_DATA();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-03-23 16:14:03 +01:00
|
|
|
/**
|
2006-07-13 16:24:55 +02:00
|
|
|
Stops the scheduler (again). Waits for acknowledgement from the
|
|
|
|
scheduler that it has stopped - synchronous stopping.
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2007-04-05 13:24:34 +02:00
|
|
|
Already running events will not be stopped. If the user needs
|
|
|
|
them stopped manual intervention is needed.
|
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
SYNOPSIS
|
2006-07-13 16:24:55 +02:00
|
|
|
Event_scheduler::stop()
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
RETURN VALUE
|
2006-07-13 16:24:55 +02:00
|
|
|
FALSE OK
|
|
|
|
TRUE Error (not reported)
|
2006-07-12 10:37:30 +02:00
|
|
|
*/
|
|
|
|
|
2006-07-13 16:24:55 +02:00
|
|
|
bool
|
|
|
|
Event_scheduler::stop()
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2006-07-13 16:24:55 +02:00
|
|
|
THD *thd= current_thd;
|
|
|
|
DBUG_ENTER("Event_scheduler::stop");
|
2006-11-27 00:47:38 +01:00
|
|
|
DBUG_PRINT("enter", ("thd: 0x%lx", (long) thd));
|
2006-07-13 16:24:55 +02:00
|
|
|
|
|
|
|
LOCK_DATA();
|
2006-11-27 00:47:38 +01:00
|
|
|
DBUG_PRINT("info", ("state before action %s", scheduler_states_names[state].str));
|
2006-07-13 16:24:55 +02:00
|
|
|
if (state != RUNNING)
|
2009-10-13 20:29:28 +02:00
|
|
|
{
|
|
|
|
/* Synchronously wait until the scheduler stops. */
|
|
|
|
while (state != INITIALIZED)
|
|
|
|
COND_STATE_WAIT(thd, NULL, "Waiting for the scheduler to stop");
|
2006-07-13 16:24:55 +02:00
|
|
|
goto end;
|
2009-10-13 20:29:28 +02:00
|
|
|
}
|
2006-07-13 16:24:55 +02:00
|
|
|
|
|
|
|
/* Guarantee we don't catch spurious signals */
|
|
|
|
do {
|
2007-03-23 17:14:08 +01:00
|
|
|
DBUG_PRINT("info", ("Waiting for COND_started_or_stopped from "
|
|
|
|
"the scheduler thread. Current value of state is %s . "
|
2006-07-13 16:24:55 +02:00
|
|
|
"workers count=%d", scheduler_states_names[state].str,
|
|
|
|
workers_count()));
|
2006-08-31 17:18:39 +02:00
|
|
|
/*
|
|
|
|
NOTE: We don't use kill_one_thread() because it can't kill COM_DEAMON
|
|
|
|
threads. In addition, kill_one_thread() requires THD but during shutdown
|
|
|
|
current_thd is NULL. Hence, if kill_one_thread should be used it has to
|
|
|
|
be modified to kill also daemons, by adding a flag, and also we have to
|
|
|
|
create artificial THD here. To save all this work, we just do what
|
|
|
|
kill_one_thread() does to kill a thread. See also sql_repl.cc for similar
|
|
|
|
usage.
|
|
|
|
*/
|
|
|
|
|
|
|
|
state= STOPPING;
|
2007-03-23 17:14:08 +01:00
|
|
|
DBUG_PRINT("info", ("Scheduler thread has id %lu",
|
|
|
|
scheduler_thd->thread_id));
|
2006-08-31 17:18:39 +02:00
|
|
|
/* Lock from delete */
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_mutex_lock(&scheduler_thd->LOCK_thd_data);
|
2006-08-31 17:18:39 +02:00
|
|
|
/* This will wake up the thread if it waits on Queue's conditional */
|
2007-03-23 17:14:08 +01:00
|
|
|
sql_print_information("Event Scheduler: Killing the scheduler thread, "
|
|
|
|
"thread id %lu",
|
2006-08-31 17:18:39 +02:00
|
|
|
scheduler_thd->thread_id);
|
|
|
|
scheduler_thd->awake(THD::KILL_CONNECTION);
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_mutex_unlock(&scheduler_thd->LOCK_thd_data);
|
2006-08-31 17:18:39 +02:00
|
|
|
|
2006-07-13 16:24:55 +02:00
|
|
|
/* thd could be 0x0, when shutting down */
|
2007-03-23 17:14:08 +01:00
|
|
|
sql_print_information("Event Scheduler: "
|
|
|
|
"Waiting for the scheduler thread to reply");
|
2006-07-17 16:52:45 +02:00
|
|
|
COND_STATE_WAIT(thd, NULL, "Waiting scheduler to stop");
|
2006-07-13 16:24:55 +02:00
|
|
|
} while (state == STOPPING);
|
2007-03-23 17:14:08 +01:00
|
|
|
DBUG_PRINT("info", ("Scheduler thread has cleaned up. Set state to INIT"));
|
2007-04-06 17:44:14 +02:00
|
|
|
sql_print_information("Event Scheduler: Stopped");
|
2006-07-13 16:24:55 +02:00
|
|
|
end:
|
|
|
|
UNLOCK_DATA();
|
|
|
|
DBUG_RETURN(FALSE);
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Returns the number of living event worker threads.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::workers_count()
|
|
|
|
*/
|
|
|
|
|
|
|
|
uint
|
|
|
|
Event_scheduler::workers_count()
|
|
|
|
{
|
|
|
|
THD *tmp;
|
|
|
|
uint count= 0;
|
2007-03-23 16:14:03 +01:00
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_ENTER("Event_scheduler::workers_count");
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_lock(&LOCK_thread_count); // For unlink from list
|
2006-07-12 10:37:30 +02:00
|
|
|
I_List_iterator<THD> it(threads);
|
|
|
|
while ((tmp=it++))
|
|
|
|
if (tmp->system_thread == SYSTEM_THREAD_EVENT_WORKER)
|
|
|
|
++count;
|
2010-01-12 02:47:27 +01:00
|
|
|
mysql_mutex_unlock(&LOCK_thread_count);
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_PRINT("exit", ("%d", count));
|
|
|
|
DBUG_RETURN(count);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Auxiliary function for locking LOCK_scheduler_state. Used
|
2006-07-13 16:24:55 +02:00
|
|
|
by the LOCK_DATA macro.
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::lock_data()
|
|
|
|
func Which function is requesting mutex lock
|
|
|
|
line On which line mutex lock is requested
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
Event_scheduler::lock_data(const char *func, uint line)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Event_scheduler::lock_data");
|
|
|
|
DBUG_PRINT("enter", ("func=%s line=%u", func, line));
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_mutex_lock(&LOCK_scheduler_state);
|
2006-07-12 10:37:30 +02:00
|
|
|
mutex_last_locked_in_func= func;
|
|
|
|
mutex_last_locked_at_line= line;
|
|
|
|
mutex_scheduler_data_locked= TRUE;
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Auxiliary function for unlocking LOCK_scheduler_state. Used
|
2006-07-13 16:24:55 +02:00
|
|
|
by the UNLOCK_DATA macro.
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::unlock_data()
|
|
|
|
func Which function is requesting mutex unlock
|
|
|
|
line On which line mutex unlock is requested
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
Event_scheduler::unlock_data(const char *func, uint line)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Event_scheduler::unlock_data");
|
|
|
|
DBUG_PRINT("enter", ("func=%s line=%u", func, line));
|
|
|
|
mutex_last_unlocked_at_line= line;
|
|
|
|
mutex_scheduler_data_locked= FALSE;
|
|
|
|
mutex_last_unlocked_in_func= func;
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_mutex_unlock(&LOCK_scheduler_state);
|
2006-07-12 10:37:30 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2010-01-12 02:47:27 +01:00
|
|
|
Wrapper for mysql_cond_wait/timedwait
|
2006-07-12 10:37:30 +02:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::cond_wait()
|
2006-07-17 16:52:45 +02:00
|
|
|
thd Thread (Could be NULL during shutdown procedure)
|
2010-01-12 02:47:27 +01:00
|
|
|
abstime If not null then call mysql_cond_timedwait()
|
2006-08-31 17:18:39 +02:00
|
|
|
msg Message for thd->proc_info
|
2006-07-17 16:52:45 +02:00
|
|
|
func Which function is requesting cond_wait
|
|
|
|
line On which line cond_wait is requested
|
2006-07-12 10:37:30 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
2006-07-17 16:52:45 +02:00
|
|
|
Event_scheduler::cond_wait(THD *thd, struct timespec *abstime, const char* msg,
|
|
|
|
const char *func, uint line)
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
2006-07-17 16:52:45 +02:00
|
|
|
DBUG_ENTER("Event_scheduler::cond_wait");
|
2006-07-12 10:37:30 +02:00
|
|
|
waiting_on_cond= TRUE;
|
|
|
|
mutex_last_unlocked_at_line= line;
|
|
|
|
mutex_scheduler_data_locked= FALSE;
|
|
|
|
mutex_last_unlocked_in_func= func;
|
2006-07-17 16:52:45 +02:00
|
|
|
if (thd)
|
|
|
|
thd->enter_cond(&COND_state, &LOCK_scheduler_state, msg);
|
|
|
|
|
2010-01-12 02:47:27 +01:00
|
|
|
DBUG_PRINT("info", ("mysql_cond_%swait", abstime? "timed":""));
|
2006-07-13 16:24:55 +02:00
|
|
|
if (!abstime)
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_cond_wait(&COND_state, &LOCK_scheduler_state);
|
2006-07-13 16:24:55 +02:00
|
|
|
else
|
2010-01-07 06:42:07 +01:00
|
|
|
mysql_cond_timedwait(&COND_state, &LOCK_scheduler_state, abstime);
|
2006-07-17 16:52:45 +02:00
|
|
|
if (thd)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
This will free the lock so we need to relock. Not the best thing to
|
|
|
|
do but we need to obey cond_wait()
|
|
|
|
*/
|
|
|
|
thd->exit_cond("");
|
|
|
|
LOCK_DATA();
|
|
|
|
}
|
2006-07-12 10:37:30 +02:00
|
|
|
mutex_last_locked_in_func= func;
|
|
|
|
mutex_last_locked_at_line= line;
|
|
|
|
mutex_scheduler_data_locked= TRUE;
|
|
|
|
waiting_on_cond= FALSE;
|
2006-07-17 16:52:45 +02:00
|
|
|
DBUG_VOID_RETURN;
|
2006-07-13 16:24:55 +02:00
|
|
|
}
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2006-07-13 16:24:55 +02:00
|
|
|
|
2006-07-12 10:37:30 +02:00
|
|
|
/*
|
|
|
|
Dumps the internal status of the scheduler
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Event_scheduler::dump_internal_status()
|
|
|
|
*/
|
|
|
|
|
2006-09-12 12:26:12 +02:00
|
|
|
void
|
|
|
|
Event_scheduler::dump_internal_status()
|
2006-07-12 10:37:30 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Event_scheduler::dump_internal_status");
|
|
|
|
|
2006-09-12 12:26:12 +02:00
|
|
|
puts("");
|
|
|
|
puts("Event scheduler status:");
|
|
|
|
printf("State : %s\n", scheduler_states_names[state].str);
|
|
|
|
printf("Thread id : %lu\n", scheduler_thd? scheduler_thd->thread_id : 0);
|
|
|
|
printf("LLA : %s:%u\n", mutex_last_locked_in_func,
|
|
|
|
mutex_last_locked_at_line);
|
|
|
|
printf("LUA : %s:%u\n", mutex_last_unlocked_in_func,
|
|
|
|
mutex_last_unlocked_at_line);
|
|
|
|
printf("WOC : %s\n", waiting_on_cond? "YES":"NO");
|
|
|
|
printf("Workers : %u\n", workers_count());
|
2006-11-27 17:16:08 +01:00
|
|
|
printf("Executed : %lu\n", (ulong) started_events);
|
2006-09-12 12:26:12 +02:00
|
|
|
printf("Data locked: %s\n", mutex_scheduler_data_locked ? "YES":"NO");
|
2006-07-12 10:37:30 +02:00
|
|
|
|
2006-09-12 12:26:12 +02:00
|
|
|
DBUG_VOID_RETURN;
|
2006-07-12 10:37:30 +02:00
|
|
|
}
|
2007-08-15 17:08:44 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
@} (End of group Event_Scheduler)
|
|
|
|
*/
|