22
22
#include "drizzled/pthread_globals.h"
23
23
#include "drizzled/internal/my_pthread.h"
24
24
#include "drizzled/internal/my_sys.h"
25
#include "drizzled/plugin/daemon.h"
26
#include "drizzled/signal_handler.h"
28
#include "drizzled/session/cache.h"
30
#include "drizzled/drizzled.h"
32
#include <boost/thread/thread.hpp>
33
#include <boost/filesystem.hpp>
26
35
#include <sys/stat.h>
30
39
static bool kill_in_progress= false;
31
static bool volatile signal_thread_in_use= false;
32
extern "C" pthread_handler_t signal_hand(void *);
40
void signal_hand(void);
36
44
extern int cleanup_done;
37
45
extern bool volatile abort_loop;
38
46
extern bool volatile shutdown_in_progress;
39
extern char pidfile_name[FN_REFLEN];
47
extern boost::filesystem::path pid_file;
40
48
/* Prototypes -> all of these should be factored out into a propper shutdown */
41
49
extern void close_connections(void);
42
50
extern std::bitset<12> test_flags;
57
66
or stop, we just want to kill the server.
60
static void kill_server(void *sig_ptr)
69
static void kill_server(int sig)
62
int sig=(int) (long) sig_ptr; // This is passed a int
63
71
// if there is a signal during the kill in progress, ignore the other
64
72
if (kill_in_progress) // Safety
66
74
kill_in_progress=true;
67
75
abort_loop=1; // This should be set
68
76
if (sig != 0) // 0 is not a valid signal number
69
my_sigset(sig, SIG_IGN); /* purify inspected */
77
ignore_signal(sig); /* purify inspected */
70
78
if (sig == SIGTERM || sig == 0)
71
79
errmsg_printf(ERRMSG_LVL_INFO, _(ER(ER_NORMAL_SHUTDOWN)),internal::my_progname);
73
81
errmsg_printf(ERRMSG_LVL_ERROR, _(ER(ER_GOT_SIGNAL)),internal::my_progname,sig);
75
82
close_connections();
76
if (sig != SIGTERM && sig != 0)
102
105
(void)close(file); /* We can ignore the error, since we are going to error anyway at this point */
104
snprintf(buff, 1024, "Can't start server: can't create PID file (%s)", pidfile_name);
107
memset(buff, 0, sizeof(buff));
108
snprintf(buff, sizeof(buff)-1, "Can't start server: can't create PID file (%s)", pid_file.file_string().c_str());
105
109
sql_perror(buff);
110
114
/** This threads handles all signals and alarms. */
111
pthread_handler_t signal_hand(void *)
115
119
internal::my_thread_init(); // Init new thread
120
boost::this_thread::at_thread_exit(&internal::my_thread_end);
116
121
signal_thread_in_use= true;
118
if (internal::thd_lib_detected != THD_LIB_LT &&
119
(test_flags.test(TEST_SIGINT)))
123
if ((test_flags.test(TEST_SIGINT)))
121
125
(void) sigemptyset(&set); // Setup up SIGINT for debug
122
126
(void) sigaddset(&set,SIGINT); // For debugging
125
129
(void) sigemptyset(&set); // Setup up SIGINT for debug
126
130
#ifndef IGNORE_SIGHUP_SIGQUIT
127
(void) sigaddset(&set,SIGQUIT);
128
(void) sigaddset(&set,SIGHUP);
131
if (sigaddset(&set,SIGQUIT))
133
std::cerr << "failed setting sigaddset() with SIGQUIT\n";
135
if (sigaddset(&set,SIGHUP))
137
std::cerr << "failed setting sigaddset() with SIGHUP\n";
130
(void) sigaddset(&set,SIGTERM);
131
(void) sigaddset(&set,SIGTSTP);
140
if (sigaddset(&set,SIGTERM))
142
std::cerr << "failed setting sigaddset() with SIGTERM\n";
144
if (sigaddset(&set,SIGTSTP))
146
std::cerr << "failed setting sigaddset() with SIGTSTP\n";
133
149
/* Save pid to this process (or thread on Linux) */
134
150
create_pid_file();
136
#ifdef HAVE_STACK_TRACE_ON_SEGV
139
sprintf(pstack_file_name,"drizzled-%lu-%%d-%%d.backtrace", (uint32_t)getpid());
140
pstack_install_segv_action(pstack_file_name);
142
#endif /* HAVE_STACK_TRACE_ON_SEGV */
145
153
signal to init that we are ready
146
154
This works by waiting for init to free mutex,
153
161
(Asked MontyW over the phone about this.) -Brian
156
if (pthread_mutex_lock(&LOCK_thread_count) == 0)
157
(void) pthread_mutex_unlock(&LOCK_thread_count);
158
(void) pthread_cond_broadcast(&COND_thread_count);
160
(void) pthread_sigmask(SIG_BLOCK,&set,NULL);
164
session::Cache::singleton().mutex().lock();
165
session::Cache::singleton().mutex().unlock();
166
COND_thread_count.notify_all();
168
if (pthread_sigmask(SIG_BLOCK, &set, NULL))
170
std::cerr << "Failed to set pthread_sigmask() in signal handler\n";
163
175
int error; // Used when debugging
164
177
if (shutdown_in_progress && !abort_loop)
170
while ((error= sigwait(&set,&sig)) == EINTR) ;
184
while ((error= sigwait(&set, &sig)) == EINTR) ;
171
187
if (cleanup_done)
173
internal::my_thread_end();
174
189
signal_thread_in_use= false;
182
198
/* switch to the old log message processing */
185
201
abort_loop=1; // mark abort for threads
186
kill_server((void*) sig); // MIT THREAD has a alarm thread
202
kill_server(sig); // MIT THREAD has a alarm thread
203
static int init(drizzled::plugin::Registry&)
206
pthread_attr_t thr_attr;
207
size_t my_thread_stack_size= 65536;
209
(void) pthread_attr_init(&thr_attr);
210
pthread_attr_setscope(&thr_attr, PTHREAD_SCOPE_SYSTEM);
211
(void) pthread_attr_setdetachstate(&thr_attr, PTHREAD_CREATE_DETACHED);
213
struct sched_param tmp_sched_param;
215
memset(&tmp_sched_param, 0, sizeof(tmp_sched_param));
216
tmp_sched_param.sched_priority= INTERRUPT_PRIOR;
217
(void)pthread_attr_setschedparam(&thr_attr, &tmp_sched_param);
219
#if defined(__ia64__) || defined(__ia64)
221
Peculiar things with ia64 platforms - it seems we only have half the
222
stack size in reality, so we have to double it here
224
pthread_attr_setstacksize(&thr_attr, my_thread_stack_size*2);
226
pthread_attr_setstacksize(&thr_attr, my_thread_stack_size);
229
(void) pthread_mutex_lock(&LOCK_thread_count);
230
if ((error=pthread_create(&signal_thread, &thr_attr, signal_hand, 0)))
232
errmsg_printf(ERRMSG_LVL_ERROR, _("Can't create interrupt-thread (error %d, errno: %d)"),
236
(void) pthread_cond_wait(&COND_thread_count,&LOCK_thread_count);
237
pthread_mutex_unlock(&LOCK_thread_count);
239
(void) pthread_attr_destroy(&thr_attr);
245
This is mainly needed when running with purify, but it's still nice to
246
know that all child threads have died when drizzled exits.
248
static int deinit(drizzled::plugin::Registry&)
252
Wait up to 10 seconds for signal thread to die. We use this mainly to
253
avoid getting warnings that internal::my_thread_end has not been called
255
for (i= 0 ; i < 100 && signal_thread_in_use; i++)
257
if (pthread_kill(signal_thread, SIGTERM) != ESRCH)
259
usleep(100); // Give it time to die
218
class SignalHandler :
219
public drizzled::plugin::Daemon
221
SignalHandler(const SignalHandler &);
222
SignalHandler& operator=(const SignalHandler &);
223
boost::thread thread;
227
drizzled::plugin::Daemon("Signal Handler")
229
// @todo fix spurious wakeup issue
230
boost::mutex::scoped_lock scopedLock(session::Cache::singleton().mutex());
231
thread= boost::thread(signal_hand);
232
signal_thread= thread.native_handle();
233
COND_thread_count.wait(scopedLock);
237
This is mainly needed when running with purify, but it's still nice to
238
know that all child threads have died when drizzled exits.
243
Wait up to 100000 micro-seconds for signal thread to die. We use this mainly to
244
avoid getting warnings that internal::my_thread_end has not been called
246
bool completed= false;
248
* We send SIGTERM and then do a timed join. If that fails we will on
249
* the last pthread_kill() call SIGTSTP. OSX (and FreeBSD) seem to
250
* prefer this. -Brian
252
uint32_t count= 2; // How many times to try join and see if the caller died.
253
while (not completed and count--)
256
int signal= count == 1 ? SIGTSTP : SIGTERM;
258
if ((error= pthread_kill(thread.native_handle(), signal)))
260
char buffer[1024]; // No reason for number;
261
strerror_r(error, buffer, sizeof(buffer));
262
std::cerr << "pthread_kill() error on shutdown of signal thread (" << buffer << ")\n";
267
boost::posix_time::milliseconds duration(100);
268
completed= thread.timed_join(duration);
274
static int init(drizzled::module::Context& context)
276
context.add(new SignalHandler);
265
282
static drizzle_sys_var* system_variables[]= {