2005-12-08 00:17:05 +01:00
|
|
|
/* Copyright (C) 2004-2005 MySQL AB
|
2005-12-05 11:45:04 +01: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
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
|
|
|
|
#include "event_priv.h"
|
2005-12-12 21:19:19 +01:00
|
|
|
#include "event.h"
|
2005-12-05 11:45:04 +01:00
|
|
|
#include "sp.h"
|
|
|
|
|
2005-12-08 20:37:54 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
Make this define DBUG_FAULTY_THR to be able to put breakpoints inside
|
|
|
|
code used by the scheduler's thread(s). In this case user connections
|
|
|
|
are not possible because the scheduler thread code is ran inside the
|
|
|
|
main thread (no spawning takes place. If you want to debug client
|
|
|
|
connection then start with --one-thread and make the define
|
2005-12-13 13:21:11 +01:00
|
|
|
DBUG_FAULTY_THR !
|
2005-12-08 20:37:54 +01:00
|
|
|
*/
|
2005-12-05 11:45:04 +01:00
|
|
|
#define DBUG_FAULTY_THR2
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
extern ulong thread_created;
|
2005-12-15 14:12:28 +01:00
|
|
|
extern const char *my_localhost;
|
2006-01-31 23:41:21 +01:00
|
|
|
extern pthread_attr_t connection_attrib;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
pthread_mutex_t LOCK_event_arrays,
|
|
|
|
LOCK_workers_count,
|
|
|
|
LOCK_evex_running;
|
|
|
|
|
|
|
|
|
|
|
|
bool evex_is_running= false;
|
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
ulonglong evex_main_thread_id= 0;
|
2005-12-05 11:45:04 +01:00
|
|
|
ulong opt_event_executor;
|
2006-01-31 23:41:21 +01:00
|
|
|
my_bool event_executor_running_global_var;
|
2005-12-06 16:15:29 +01:00
|
|
|
static my_bool evex_mutexes_initted= false;
|
2005-12-07 22:29:00 +01:00
|
|
|
static uint workers_count;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
static int
|
|
|
|
evex_load_events_from_db(THD *thd);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
TODO Andrey: Check for command line option whether to start
|
|
|
|
the main thread or not.
|
|
|
|
*/
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
pthread_handler_t
|
|
|
|
event_executor_worker(void *arg);
|
|
|
|
|
|
|
|
pthread_handler_t
|
|
|
|
event_executor_main(void *arg);
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-28 12:07:57 +02:00
|
|
|
static int
|
|
|
|
evex_time_diff(TIME *a, TIME *b)
|
|
|
|
{
|
|
|
|
return sec_since_epoch_TIME(a) - sec_since_epoch_TIME(b);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
evex_init_mutexes()
|
2005-12-06 16:15:29 +01:00
|
|
|
{
|
|
|
|
if (evex_mutexes_initted)
|
|
|
|
return;
|
2005-12-13 23:10:29 +01:00
|
|
|
|
|
|
|
evex_mutexes_initted= true;
|
2005-12-06 16:15:29 +01:00
|
|
|
pthread_mutex_init(&LOCK_event_arrays, MY_MUTEX_INIT_FAST);
|
|
|
|
pthread_mutex_init(&LOCK_workers_count, MY_MUTEX_INIT_FAST);
|
|
|
|
pthread_mutex_init(&LOCK_evex_running, MY_MUTEX_INIT_FAST);
|
2005-12-13 23:10:29 +01:00
|
|
|
|
|
|
|
event_executor_running_global_var= opt_event_executor;
|
2005-12-06 16:15:29 +01:00
|
|
|
}
|
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
int
|
|
|
|
init_events()
|
|
|
|
{
|
|
|
|
pthread_t th;
|
|
|
|
|
|
|
|
DBUG_ENTER("init_events");
|
|
|
|
|
|
|
|
DBUG_PRINT("info",("Starting events main thread"));
|
|
|
|
|
2005-12-06 16:15:29 +01:00
|
|
|
evex_init_mutexes();
|
2005-12-12 21:19:19 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
VOID(pthread_mutex_lock(&LOCK_evex_running));
|
|
|
|
evex_is_running= false;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_evex_running));
|
2005-12-20 14:21:02 +02:00
|
|
|
|
|
|
|
if (event_executor_running_global_var)
|
|
|
|
{
|
2005-12-05 11:45:04 +01:00
|
|
|
#ifndef DBUG_FAULTY_THR
|
2005-12-20 14:21:02 +02:00
|
|
|
//TODO Andrey: Change the error code returned!
|
2006-01-31 23:41:21 +01:00
|
|
|
if (pthread_create(&th, &connection_attrib, event_executor_main,(void*)NULL))
|
2005-12-20 14:21:02 +02:00
|
|
|
DBUG_RETURN(ER_SLAVE_THREAD);
|
2005-12-05 11:45:04 +01:00
|
|
|
#else
|
2005-12-20 14:21:02 +02:00
|
|
|
event_executor_main(NULL);
|
2005-12-05 11:45:04 +01:00
|
|
|
#endif
|
2005-12-20 14:21:02 +02:00
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
shutdown_events()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("shutdown_events");
|
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_evex_running));
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_evex_running));
|
2005-12-12 21:19:19 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
pthread_mutex_destroy(&LOCK_event_arrays);
|
|
|
|
pthread_mutex_destroy(&LOCK_workers_count);
|
|
|
|
pthread_mutex_destroy(&LOCK_evex_running);
|
|
|
|
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
init_event_thread(THD* thd)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("init_event_thread");
|
|
|
|
thd->client_capabilities= 0;
|
|
|
|
thd->security_ctx->skip_grants();
|
2005-12-15 14:12:28 +01:00
|
|
|
thd->security_ctx->host= (char*)my_localhost;
|
2005-12-05 11:45:04 +01:00
|
|
|
my_net_init(&thd->net, 0);
|
|
|
|
thd->net.read_timeout = slave_net_timeout;
|
|
|
|
thd->slave_thread= 0;
|
|
|
|
thd->options= OPTION_AUTO_IS_NULL;
|
|
|
|
thd->client_capabilities= CLIENT_LOCAL_FILES;
|
|
|
|
thd->real_id=pthread_self();
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_thread_count));
|
|
|
|
thd->thread_id= thread_id++;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_thread_count));
|
|
|
|
|
|
|
|
if (init_thr_lock() || thd->store_globals())
|
|
|
|
{
|
|
|
|
thd->cleanup();
|
|
|
|
delete thd;
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if !defined(__WIN__) && !defined(OS2) && !defined(__NETWARE__)
|
|
|
|
sigset_t set;
|
|
|
|
VOID(sigemptyset(&set)); // Get mask in use
|
|
|
|
VOID(pthread_sigmask(SIG_UNBLOCK,&set,&thd->block_signals));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
thd->proc_info= "Initialized";
|
|
|
|
thd->version= refresh_version;
|
|
|
|
thd->set_time();
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
pthread_handler_t
|
|
|
|
event_executor_main(void *arg)
|
2005-12-05 11:45:04 +01:00
|
|
|
{
|
|
|
|
THD *thd; /* needs to be first for thread_stack */
|
|
|
|
ulonglong iter_num= 0;
|
|
|
|
uint i=0, j=0;
|
2005-12-13 19:16:00 +01:00
|
|
|
my_ulonglong cnt= 0;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
DBUG_ENTER("event_executor_main");
|
|
|
|
DBUG_PRINT("event_executor_main", ("EVEX thread started"));
|
|
|
|
|
|
|
|
|
|
|
|
// init memory root
|
|
|
|
init_alloc_root(&evex_mem_root, MEM_ROOT_BLOCK_SIZE, MEM_ROOT_PREALLOC);
|
2005-12-07 22:29:00 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
// needs to call my_thread_init(), otherwise we get a coredump in DBUG_ stuff
|
|
|
|
my_thread_init();
|
2006-01-12 16:51:45 +01:00
|
|
|
|
|
|
|
if (sizeof(my_time_t) != sizeof(time_t))
|
|
|
|
{
|
|
|
|
sql_print_error("sizeof(my_time_t) != sizeof(time_t) ."
|
|
|
|
"The scheduler will not work correctly. Stopping.");
|
|
|
|
goto err_no_thd;
|
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
//TODO Andrey: Check for NULL
|
|
|
|
if (!(thd = new THD)) // note that contructor of THD uses DBUG_ !
|
|
|
|
{
|
|
|
|
sql_print_error("Cannot create THD for event_executor_main");
|
|
|
|
goto err_no_thd;
|
2006-01-12 16:51:45 +01:00
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
thd->thread_stack = (char*)&thd; // remember where our stack is
|
|
|
|
|
|
|
|
pthread_detach_this_thread();
|
|
|
|
|
|
|
|
if (init_event_thread(thd))
|
|
|
|
goto err;
|
2005-12-12 21:19:19 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
// make this thread invisible it has no vio -> show processlist won't see
|
2005-12-06 16:15:29 +01:00
|
|
|
thd->system_thread= 1;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_thread_count));
|
|
|
|
threads.append(thd);
|
|
|
|
thread_count++;
|
|
|
|
thread_running++;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_thread_count));
|
|
|
|
|
2005-12-13 19:16:00 +01:00
|
|
|
DBUG_PRINT("EVEX main thread", ("Initing events_queuey"));
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
/*
|
|
|
|
eventually manifest that we are running, not to crashe because of
|
|
|
|
usage of non-initialized memory structures.
|
|
|
|
*/
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_evex_running));
|
2005-12-13 19:16:00 +01:00
|
|
|
VOID(pthread_mutex_lock(&LOCK_event_arrays));
|
|
|
|
evex_queue_init(&EVEX_EQ_NAME);
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
2005-12-07 22:29:00 +01:00
|
|
|
evex_is_running= true;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_evex_running));
|
|
|
|
|
2005-12-16 12:42:39 +01:00
|
|
|
thd->security_ctx->user= my_strdup("event_scheduler", MYF(0));
|
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
if (evex_load_events_from_db(thd))
|
2005-12-08 20:37:54 +01:00
|
|
|
goto err;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
evex_main_thread_id= thd->thread_id;
|
2005-12-16 12:42:39 +01:00
|
|
|
|
2005-12-13 19:16:00 +01:00
|
|
|
sql_print_information("Scheduler thread started");
|
2005-12-05 11:45:04 +01:00
|
|
|
while (!thd->killed)
|
|
|
|
{
|
|
|
|
TIME time_now;
|
|
|
|
my_time_t now;
|
2005-12-12 21:19:19 +01:00
|
|
|
event_timed *et;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-13 19:16:00 +01:00
|
|
|
cnt++;
|
2006-01-11 18:09:05 +01:00
|
|
|
DBUG_PRINT("info", ("EVEX External Loop %d thd->k", cnt));
|
2005-12-13 23:10:29 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
thd->proc_info = "Sleeping";
|
2006-01-11 18:09:05 +01:00
|
|
|
if (!event_executor_running_global_var)
|
|
|
|
{
|
|
|
|
sql_print_information("Scheduler asked to stop.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!evex_queue_num_elements(EVEX_EQ_NAME))
|
2005-12-12 21:19:19 +01:00
|
|
|
{
|
|
|
|
my_sleep(1000000);// sleep 1s
|
2005-12-05 11:45:04 +01:00
|
|
|
continue;
|
2005-12-12 21:19:19 +01:00
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
{
|
2005-12-12 21:19:19 +01:00
|
|
|
int t2sleep;
|
|
|
|
/*
|
|
|
|
now let's see how much time to sleep, we know there is at least 1
|
|
|
|
element in the queue.
|
|
|
|
*/
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_event_arrays));
|
|
|
|
if (!evex_queue_num_elements(EVEX_EQ_NAME))
|
|
|
|
{
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
et= evex_queue_first_element(&EVEX_EQ_NAME, event_timed*);
|
2005-12-20 14:21:02 +02:00
|
|
|
if (et->status == MYSQL_EVENT_DISABLED)
|
|
|
|
{
|
2006-01-11 19:31:21 +01:00
|
|
|
DBUG_PRINT("evex main thread",("Now it is disabled-exec no more"));
|
2005-12-20 14:21:02 +02:00
|
|
|
if (et->dropped)
|
|
|
|
et->drop(thd);
|
|
|
|
delete et;
|
|
|
|
evex_queue_delete_element(&EVEX_EQ_NAME, 1);// 1 is top
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
|
|
|
sql_print_information("Event found disabled, dropping.");
|
|
|
|
continue;
|
|
|
|
}
|
2005-12-12 21:19:19 +01:00
|
|
|
|
2006-01-11 19:31:21 +01:00
|
|
|
DBUG_PRINT("evex main thread",("computing time to sleep till next exec"));
|
2006-01-12 16:51:45 +01:00
|
|
|
time((time_t *)&now);
|
2005-12-12 21:19:19 +01:00
|
|
|
my_tz_UTC->gmt_sec_to_TIME(&time_now, now);
|
|
|
|
t2sleep= evex_time_diff(&et->execute_at, &time_now);
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
2006-01-11 19:31:21 +01:00
|
|
|
|
|
|
|
DBUG_PRINT("evex main thread",("unlocked LOCK_event_arrays"));
|
2005-12-12 21:19:19 +01:00
|
|
|
if (t2sleep > 0)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
We sleep t2sleep seconds but we check every second whether this thread
|
2005-12-28 12:07:57 +02:00
|
|
|
has been killed, or there is a new candidate
|
2005-12-12 21:19:19 +01:00
|
|
|
*/
|
2006-01-11 18:09:05 +01:00
|
|
|
while (t2sleep-- && !thd->killed && event_executor_running_global_var &&
|
2005-12-12 21:19:19 +01:00
|
|
|
evex_queue_num_elements(EVEX_EQ_NAME) &&
|
|
|
|
(evex_queue_first_element(&EVEX_EQ_NAME, event_timed*) == et))
|
2006-01-11 19:31:21 +01:00
|
|
|
{
|
|
|
|
DBUG_PRINT("evex main thread",("will sleep a bit more"));
|
2005-12-12 21:19:19 +01:00
|
|
|
my_sleep(1000000);
|
2006-01-11 19:31:21 +01:00
|
|
|
}
|
2005-12-12 21:19:19 +01:00
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
if (!event_executor_running_global_var)
|
2006-01-11 18:09:05 +01:00
|
|
|
{
|
|
|
|
sql_print_information("Scheduler asked to stop.");
|
|
|
|
break;
|
|
|
|
}
|
2005-12-12 21:19:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_event_arrays));
|
|
|
|
|
|
|
|
if (!evex_queue_num_elements(EVEX_EQ_NAME))
|
|
|
|
{
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
2006-01-11 19:31:21 +01:00
|
|
|
DBUG_PRINT("evex main thread",("empty queue"));
|
2005-12-12 21:19:19 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
et= evex_queue_first_element(&EVEX_EQ_NAME, event_timed*);
|
2006-01-11 19:31:21 +01:00
|
|
|
DBUG_PRINT("evex main thread",("got event from the queue"));
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
if (et->execute_at.year > 1969 &&
|
|
|
|
my_time_compare(&time_now, &et->execute_at) == -1)
|
|
|
|
{
|
2006-01-11 19:31:21 +01:00
|
|
|
DBUG_PRINT("evex main thread",("still not the time for execution"));
|
2005-12-12 21:19:19 +01:00
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2006-01-11 19:31:21 +01:00
|
|
|
DBUG_PRINT("evex main thread",("it's right time"));
|
2005-12-12 21:19:19 +01:00
|
|
|
if (et->status == MYSQL_EVENT_ENABLED)
|
|
|
|
{
|
|
|
|
pthread_t th;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2006-01-18 20:41:22 +01:00
|
|
|
DBUG_PRINT("evex main thread", ("[%10s] this exec at [%llu]", et->name.str,
|
|
|
|
TIME_to_ulonglong_datetime(&et->execute_at)));
|
|
|
|
et->mark_last_executed(thd);
|
|
|
|
if (et->compute_next_execution_time())
|
|
|
|
{
|
|
|
|
sql_print_error("Error while computing time of %s.%s . "
|
|
|
|
"Disabling after execution.",
|
|
|
|
et->dbname.str, et->name.str);
|
|
|
|
et->status= MYSQL_EVENT_DISABLED;
|
|
|
|
}
|
|
|
|
DBUG_PRINT("evex main thread", ("[%10s] next exec at [%llu]", et->name.str,
|
|
|
|
TIME_to_ulonglong_datetime(&et->execute_at)));
|
|
|
|
|
2005-12-28 12:07:57 +02:00
|
|
|
et->update_fields(thd);
|
2006-01-30 13:15:23 +01:00
|
|
|
++iter_num;
|
|
|
|
DBUG_PRINT("info", (" Spawning a thread %d", iter_num));
|
2005-12-05 11:45:04 +01:00
|
|
|
#ifndef DBUG_FAULTY_THR
|
2006-01-31 23:41:21 +01:00
|
|
|
if (pthread_create(&th,&connection_attrib,event_executor_worker,(void*)et))
|
2005-12-12 21:19:19 +01:00
|
|
|
{
|
|
|
|
sql_print_error("Problem while trying to create a thread");
|
|
|
|
UNLOCK_MUTEX_AND_BAIL_OUT(LOCK_event_arrays, err);
|
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
#else
|
2005-12-12 21:19:19 +01:00
|
|
|
event_executor_worker((void *) et);
|
2005-12-05 11:45:04 +01:00
|
|
|
#endif
|
2005-12-12 21:19:19 +01:00
|
|
|
if ((et->execute_at.year && !et->expression) ||
|
2005-12-20 14:21:02 +02:00
|
|
|
TIME_to_ulonglong_datetime(&et->execute_at) == 0)
|
2005-12-12 21:19:19 +01:00
|
|
|
et->flags |= EVENT_EXEC_NO_MORE;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-28 12:07:57 +02:00
|
|
|
if ((et->flags & EVENT_EXEC_NO_MORE) || et->status == MYSQL_EVENT_DISABLED)
|
|
|
|
evex_queue_delete_element(&EVEX_EQ_NAME, 1);// 1 is top
|
|
|
|
else
|
|
|
|
evex_queue_first_updated(&EVEX_EQ_NAME);
|
|
|
|
}
|
2006-01-11 19:31:21 +01:00
|
|
|
DBUG_PRINT("evex main thread",("unlocking"));
|
2005-12-05 11:45:04 +01:00
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
2005-12-12 21:19:19 +01:00
|
|
|
}// while
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
err:
|
2005-12-07 22:29:00 +01:00
|
|
|
// First manifest that this thread does not work and then destroy
|
2005-12-05 11:45:04 +01:00
|
|
|
VOID(pthread_mutex_lock(&LOCK_evex_running));
|
2005-12-12 21:19:19 +01:00
|
|
|
evex_is_running= false;
|
|
|
|
evex_main_thread_id= 0;
|
2005-12-05 11:45:04 +01:00
|
|
|
VOID(pthread_mutex_unlock(&LOCK_evex_running));
|
|
|
|
|
2006-01-11 18:09:05 +01:00
|
|
|
sql_print_information("Event scheduler stopping. Waiting for worker threads to finish.");
|
2005-12-07 22:29:00 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
TODO: A better will be with a conditional variable
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
Read workers_count without lock, no need for locking.
|
|
|
|
In the worst case we have to wait 1sec more.
|
|
|
|
*/
|
|
|
|
while (workers_count)
|
|
|
|
my_sleep(1000000);// 1s
|
|
|
|
|
|
|
|
/*
|
|
|
|
LEX_STRINGs reside in the memory root and will be destroyed with it.
|
|
|
|
Hence no need of delete but only freeing of SP
|
|
|
|
*/
|
2005-12-13 19:16:00 +01:00
|
|
|
// First we free all objects ...
|
|
|
|
for (i= 0; i < evex_queue_num_elements(EVEX_EQ_NAME); ++i)
|
|
|
|
{
|
|
|
|
event_timed *et= evex_queue_element(&EVEX_EQ_NAME, i, event_timed*);
|
|
|
|
et->free_sp();
|
|
|
|
delete et;
|
|
|
|
}
|
|
|
|
// ... then we can thras the whole queue at once
|
2005-12-12 21:19:19 +01:00
|
|
|
evex_queue_destroy(&EVEX_EQ_NAME);
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
thd->proc_info = "Clearing";
|
|
|
|
DBUG_ASSERT(thd->net.buff != 0);
|
|
|
|
net_end(&thd->net); // destructor will not free it, because we are weird
|
|
|
|
THD_CHECK_SENTRY(thd);
|
2005-12-07 22:29:00 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
pthread_mutex_lock(&LOCK_thread_count);
|
|
|
|
thread_count--;
|
|
|
|
thread_running--;
|
2005-12-12 21:19:19 +01:00
|
|
|
#ifndef DBUG_FAULTY_THR
|
2005-12-05 11:45:04 +01:00
|
|
|
THD_CHECK_SENTRY(thd);
|
|
|
|
delete thd;
|
2005-12-12 21:19:19 +01:00
|
|
|
#endif
|
2005-12-05 11:45:04 +01:00
|
|
|
pthread_mutex_unlock(&LOCK_thread_count);
|
|
|
|
|
|
|
|
|
|
|
|
err_no_thd:
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_evex_running));
|
|
|
|
evex_is_running= false;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_evex_running));
|
|
|
|
|
|
|
|
free_root(&evex_mem_root, MYF(0));
|
2006-01-11 18:09:05 +01:00
|
|
|
sql_print_information("Event scheduler stopped.");
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
#ifndef DBUG_FAULTY_THR
|
2005-12-05 11:45:04 +01:00
|
|
|
my_thread_end();
|
|
|
|
pthread_exit(0);
|
2005-12-12 21:19:19 +01:00
|
|
|
#endif
|
2005-12-07 22:29:00 +01:00
|
|
|
DBUG_RETURN(0);// Can't return anything here
|
2005-12-05 11:45:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
pthread_handler_t
|
|
|
|
event_executor_worker(void *event_void)
|
2005-12-05 11:45:04 +01:00
|
|
|
{
|
|
|
|
THD *thd; /* needs to be first for thread_stack */
|
|
|
|
event_timed *event = (event_timed *) event_void;
|
2005-12-07 22:29:00 +01:00
|
|
|
MEM_ROOT worker_mem_root;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
DBUG_ENTER("event_executor_worker");
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_workers_count));
|
|
|
|
++workers_count;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_workers_count));
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
init_alloc_root(&worker_mem_root, MEM_ROOT_BLOCK_SIZE, MEM_ROOT_PREALLOC);
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
#ifndef DBUG_FAULTY_THR
|
2005-12-05 11:45:04 +01:00
|
|
|
my_thread_init();
|
|
|
|
|
|
|
|
if (!(thd = new THD)) // note that contructor of THD uses DBUG_ !
|
|
|
|
{
|
2005-12-08 00:17:05 +01:00
|
|
|
sql_print_error("Cannot create a THD structure in a scheduler worker thread");
|
2005-12-05 11:45:04 +01:00
|
|
|
goto err_no_thd;
|
|
|
|
}
|
|
|
|
thd->thread_stack = (char*)&thd; // remember where our stack is
|
2005-12-07 22:29:00 +01:00
|
|
|
thd->mem_root= &worker_mem_root;
|
|
|
|
|
2006-01-30 13:09:08 +01:00
|
|
|
pthread_detach_this_thread();
|
2006-01-30 13:15:23 +01:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
if (init_event_thread(thd))
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
thd->init_for_queries();
|
|
|
|
|
|
|
|
// make this thread visible it has no vio -> show processlist needs this flag
|
2005-12-06 16:15:29 +01:00
|
|
|
thd->system_thread= 1;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_thread_count));
|
|
|
|
threads.append(thd);
|
|
|
|
thread_count++;
|
|
|
|
thread_running++;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_thread_count));
|
2005-12-12 21:19:19 +01:00
|
|
|
#else
|
|
|
|
thd= current_thd;
|
|
|
|
#endif
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
// thd->security_ctx->priv_host is char[MAX_HOSTNAME]
|
|
|
|
|
|
|
|
strxnmov(thd->security_ctx->priv_host, sizeof(thd->security_ctx->priv_host),
|
2005-12-08 00:17:05 +01:00
|
|
|
event->definer_host.str, NullS);
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-28 12:07:57 +02:00
|
|
|
thd->security_ctx->user= thd->security_ctx->priv_user=
|
|
|
|
my_strdup(event->definer_user.str, MYF(0));
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-08 00:17:05 +01:00
|
|
|
thd->db= event->dbname.str;
|
2005-12-12 21:19:19 +01:00
|
|
|
if (!check_access(thd, EVENT_ACL, event->dbname.str, 0, 0, 0,
|
|
|
|
is_schema_db(event->dbname.str)))
|
2005-12-05 11:45:04 +01:00
|
|
|
{
|
|
|
|
int ret;
|
2006-01-10 11:31:45 +01:00
|
|
|
DBUG_PRINT("info", (" EVEX EXECUTING event %s.%s [EXPR:%d]",
|
|
|
|
event->dbname.str, event->name.str,(int) event->expression));
|
|
|
|
sql_print_information(" EVEX EXECUTING event %s.%s [EXPR:%d]",
|
|
|
|
event->dbname.str, event->name.str,(int) event->expression);
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
ret= event->execute(thd, &worker_mem_root);
|
2006-01-10 11:31:45 +01:00
|
|
|
|
|
|
|
sql_print_information(" EVEX EXECUTED event %s.%s [EXPR:%d]. RetCode=%d",
|
|
|
|
event->dbname.str, event->name.str,
|
|
|
|
(int) event->expression, ret);
|
2006-01-20 22:24:58 +01:00
|
|
|
if (ret == EVEX_COMPILE_ERROR)
|
|
|
|
sql_print_information(" EVEX COMPILE ERROR for event %s.%s",
|
|
|
|
event->dbname.str, event->name.str);
|
|
|
|
|
2006-01-10 11:31:45 +01:00
|
|
|
DBUG_PRINT("info", (" EVEX EXECUTED event %s.%s [EXPR:%d]. RetCode=%d",
|
|
|
|
event->dbname.str, event->name.str,
|
|
|
|
(int) event->expression, ret));
|
2005-12-05 11:45:04 +01:00
|
|
|
}
|
2005-12-28 12:07:57 +02:00
|
|
|
if ((event->flags & EVENT_EXEC_NO_MORE) || event->status==MYSQL_EVENT_DISABLED)
|
|
|
|
{
|
2006-01-20 22:24:58 +01:00
|
|
|
DBUG_PRINT("event_executor_worker",
|
|
|
|
("%s exec no more. to drop=%d",event->name.str, event->dropped));
|
2005-12-28 12:07:57 +02:00
|
|
|
if (event->dropped)
|
|
|
|
event->drop(thd);
|
|
|
|
delete event;
|
|
|
|
}
|
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
thd->db= 0;
|
|
|
|
|
|
|
|
err:
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_thread_count));
|
2005-12-12 21:19:19 +01:00
|
|
|
#ifndef DBUG_FAULTY_THR
|
2005-12-05 11:45:04 +01:00
|
|
|
thread_count--;
|
|
|
|
thread_running--;
|
|
|
|
/*
|
|
|
|
Some extra safety, which should not been needed (normally, event deletion
|
|
|
|
should already have done these assignments (each event which sets these
|
|
|
|
variables is supposed to set them to 0 before terminating)).
|
|
|
|
*/
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_thread_count));
|
|
|
|
|
|
|
|
thd->proc_info = "Clearing";
|
|
|
|
DBUG_ASSERT(thd->net.buff != 0);
|
|
|
|
net_end(&thd->net); // destructor will not free it, because we are weird
|
|
|
|
THD_CHECK_SENTRY(thd);
|
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_thread_count));
|
|
|
|
THD_CHECK_SENTRY(thd);
|
|
|
|
delete thd;
|
2005-12-12 21:19:19 +01:00
|
|
|
#endif
|
2005-12-05 11:45:04 +01:00
|
|
|
VOID(pthread_mutex_unlock(&LOCK_thread_count));
|
|
|
|
|
|
|
|
err_no_thd:
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
free_root(&worker_mem_root, MYF(0));
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_workers_count));
|
|
|
|
--workers_count;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_workers_count));
|
|
|
|
|
|
|
|
#ifndef DBUG_FAULTY_THR
|
|
|
|
my_thread_end();
|
|
|
|
pthread_exit(0);
|
|
|
|
#endif
|
|
|
|
DBUG_RETURN(0); // Can't return anything here
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
evex_load_events_from_db(THD *thd)
|
|
|
|
{
|
|
|
|
TABLE *table;
|
|
|
|
READ_RECORD read_record_info;
|
|
|
|
MYSQL_LOCK *lock;
|
|
|
|
int ret= -1;
|
2005-12-15 14:12:28 +01:00
|
|
|
uint count= 0;
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
DBUG_ENTER("evex_load_events_from_db");
|
|
|
|
|
2005-12-16 12:42:39 +01:00
|
|
|
if ((ret= evex_open_event_table(thd, TL_READ, &table)))
|
|
|
|
{
|
|
|
|
sql_print_error("Table mysql.event is damaged.");
|
2005-12-05 11:45:04 +01:00
|
|
|
DBUG_RETURN(SP_OPEN_TABLE_FAILED);
|
2005-12-16 12:42:39 +01:00
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_event_arrays));
|
|
|
|
|
|
|
|
init_read_record(&read_record_info, thd, table ,NULL,1,0);
|
|
|
|
while (!(read_record_info.read_record(&read_record_info)))
|
|
|
|
{
|
2005-12-13 19:16:00 +01:00
|
|
|
event_timed *et;
|
2005-12-13 13:21:11 +01:00
|
|
|
if (!(et= new event_timed))
|
2005-12-05 11:45:04 +01:00
|
|
|
{
|
|
|
|
DBUG_PRINT("evex_load_events_from_db", ("Out of memory"));
|
|
|
|
ret= -1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
DBUG_PRINT("evex_load_events_from_db", ("Loading event from row."));
|
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
if ((ret= et->load_from_row(&evex_mem_root, table)))
|
|
|
|
{
|
|
|
|
sql_print_error("Error while loading from mysql.event. "
|
|
|
|
"Table probably corrupted");
|
|
|
|
goto end;
|
|
|
|
}
|
2005-12-12 21:19:19 +01:00
|
|
|
if (et->status != MYSQL_EVENT_ENABLED)
|
|
|
|
{
|
2006-01-10 11:31:45 +01:00
|
|
|
DBUG_PRINT("evex_load_events_from_db",("%s is disabled",et->name.str));
|
2005-12-12 21:19:19 +01:00
|
|
|
delete et;
|
|
|
|
continue;
|
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
DBUG_PRINT("evex_load_events_from_db",
|
2005-12-08 00:17:05 +01:00
|
|
|
("Event %s loaded from row. Time to compile", et->name.str));
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-07 22:29:00 +01:00
|
|
|
if ((ret= et->compile(thd, &evex_mem_root)))
|
|
|
|
{
|
|
|
|
sql_print_error("Error while compiling %s.%s. Aborting load.",
|
2005-12-08 00:17:05 +01:00
|
|
|
et->dbname.str, et->name.str);
|
2005-12-07 22:29:00 +01:00
|
|
|
goto end;
|
|
|
|
}
|
2005-12-20 14:21:02 +02:00
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
// let's find when to be executed
|
2006-01-18 20:41:22 +01:00
|
|
|
if (et->compute_next_execution_time())
|
|
|
|
{
|
|
|
|
sql_print_error("Error while computing execution time of %s.%s. Skipping",
|
|
|
|
et->dbname.str, et->name.str);
|
|
|
|
continue;
|
|
|
|
}
|
2005-12-05 11:45:04 +01:00
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
DBUG_PRINT("evex_load_events_from_db", ("Adding to the exec list."));
|
2005-12-13 19:16:00 +01:00
|
|
|
|
|
|
|
evex_queue_insert(&EVEX_EQ_NAME, (EVEX_PTOQEL) et);
|
|
|
|
DBUG_PRINT("evex_load_events_from_db", ("%p %*s",
|
|
|
|
et, et->name.length,et->name.str));
|
2005-12-15 14:12:28 +01:00
|
|
|
count++;
|
2005-12-05 11:45:04 +01:00
|
|
|
}
|
|
|
|
|
2005-12-12 21:19:19 +01:00
|
|
|
ret= 0;
|
|
|
|
|
|
|
|
end:
|
2005-12-05 11:45:04 +01:00
|
|
|
VOID(pthread_mutex_unlock(&LOCK_event_arrays));
|
2005-12-12 21:19:19 +01:00
|
|
|
end_read_record(&read_record_info);
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
thd->version--; // Force close to free memory
|
|
|
|
|
|
|
|
close_thread_tables(thd);
|
2006-01-10 11:31:45 +01:00
|
|
|
sql_print_information("Scheduler loaded %d event%s", count, (count == 1)?"":"s");
|
|
|
|
DBUG_PRINT("info", ("Status code %d. Loaded %d event(s)", ret, count));
|
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
DBUG_RETURN(ret);
|
|
|
|
}
|
2005-12-06 16:15:29 +01:00
|
|
|
|
2005-12-07 19:26:44 +01:00
|
|
|
|
2005-12-06 16:15:29 +01:00
|
|
|
bool sys_var_event_executor::update(THD *thd, set_var *var)
|
|
|
|
{
|
|
|
|
// here start the thread if not running.
|
2006-01-11 18:09:05 +01:00
|
|
|
DBUG_ENTER("sys_var_event_executor::update");
|
2005-12-06 16:15:29 +01:00
|
|
|
VOID(pthread_mutex_lock(&LOCK_evex_running));
|
2005-12-13 23:10:29 +01:00
|
|
|
*value= var->save_result.ulong_value;
|
2006-01-11 18:09:05 +01:00
|
|
|
|
|
|
|
DBUG_PRINT("new_value", ("%d", *value));
|
2005-12-13 23:10:29 +01:00
|
|
|
if ((my_bool) *value && !evex_is_running)
|
2005-12-07 22:29:00 +01:00
|
|
|
{
|
2005-12-06 16:15:29 +01:00
|
|
|
VOID(pthread_mutex_unlock(&LOCK_evex_running));
|
|
|
|
init_events();
|
|
|
|
} else
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_evex_running));
|
2006-01-11 18:09:05 +01:00
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
2005-12-06 16:15:29 +01:00
|
|
|
}
|
|
|
|
|