diff options
Diffstat (limited to 'sql/mysqld.cc')
-rw-r--r-- | sql/mysqld.cc | 3433 |
1 files changed, 3433 insertions, 0 deletions
diff --git a/sql/mysqld.cc b/sql/mysqld.cc new file mode 100644 index 00000000000..179c7ecd9dc --- /dev/null +++ b/sql/mysqld.cc @@ -0,0 +1,3433 @@ +/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB + + 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 "mysql_priv.h" +#include <mysql.h> +#include <m_ctype.h> +#include "sql_acl.h" +#ifdef HAVE_BERKELEY_DB +#include "ha_berkeley.h" +#endif +#include "ha_myisam.h" +#include <nisam.h> +#include <thr_alarm.h> +#include <ft_global.h> + +#ifdef __cplusplus +extern "C" { // Because of SCO 3.2V4.2 +#endif +#include <errno.h> +#include <sys/stat.h> +#ifndef __GNU_LIBRARY__ +#define __GNU_LIBRARY__ // Skipp warnings in getopt.h +#endif +#include <getopt.h> +#ifdef HAVE_SYSENT_H +#include <sysent.h> +#endif +#ifdef HAVE_PWD_H +#include <pwd.h> // For getpwent +#endif +#ifdef HAVE_GRP_H +#include <grp.h> +#endif + +#ifndef __WIN__ +#include <sys/resource.h> +#ifdef HAVE_SYS_UN_H +# include <sys/un.h> +#endif +#include <netdb.h> +#ifdef HAVE_SELECT_H +# include <select.h> +#endif +#ifdef HAVE_SYS_SELECT_H +#include <sys/select.h> +#endif +#include <sys/utsname.h> +#else +#include <windows.h> +#endif // __WIN__ + +#ifdef HAVE_LIBWRAP +#include <tcpd.h> +#include <syslog.h> +#ifdef NEED_SYS_SYSLOG_H +#include <sys/syslog.h> +#endif /* NEED_SYS_SYSLOG_H */ +int allow_severity = LOG_INFO; +int deny_severity = LOG_WARNING; +#endif /* HAVE_LIBWRAP */ + +#if defined(__FreeBSD__) && defined(HAVE_IEEEFP_H) +#include <ieeefp.h> +#ifdef HAVE_FP_EXCEPT // Fix type conflict +typedef fp_except fp_except_t; +#endif + + /* We can't handle floating point expections with threads, so disable + this on freebsd + */ + +inline void reset_floating_point_exceptions() +{ + /* Don't fall for overflow, underflow,divide-by-zero or loss of precision */ + fpsetmask(~(FP_X_INV | FP_X_DNML | FP_X_OFL | FP_X_UFL | + FP_X_DZ | FP_X_IMP)); +} +#else +#define reset_floating_point_exceptions() +#endif /* __FreeBSD__ && HAVE_IEEEFP_H */ + +#ifdef __cplusplus +} +#endif + +#if defined(HAVE_LINUXTHREADS) +#define THR_KILL_SIGNAL SIGINT +#else +#define THR_KILL_SIGNAL SIGUSR2 // Can't use this with LinuxThreads +#endif + +#ifdef HAVE_GLIBC2_STYLE_GETHOSTBYNAME_R +#include <sys/types.h> +#else +#include <my_pthread.h> // For thr_setconcurency() +#endif +#if defined(HAVE_GETRLIMIT) && defined(RLIMIT_NOFILE) && !defined(__linux__) && !defined(HAVE_mit_thread) +#define SET_RLIMIT_NOFILE +#endif + +#ifdef SOLARIS +extern "C" int gethostname(char *name, int namelen); +#endif + +#define MYSQL_KILL_SIGNAL SIGTERM + +#ifndef DBUG_OFF +static const char* default_dbug_option=IF_WIN("d:t:i:O,\\mysqld.trace", + "d:t:i:o,/tmp/mysqld.trace"); +#endif + +#ifdef __NT__ +static char szPipeName [ 257 ]; +static SECURITY_ATTRIBUTES saPipeSecurity; +static SECURITY_DESCRIPTOR sdPipeDescriptor; +static HANDLE hPipe = INVALID_HANDLE_VALUE; +static pthread_cond_t COND_handler_count; +static uint handler_count; +#endif +#ifdef __WIN__ +static bool opt_console=0; +#endif + +static ulong opt_specialflag=SPECIAL_ENGLISH; +static my_socket unix_sock= INVALID_SOCKET,ip_sock= INVALID_SOCKET; +static ulong back_log,connect_timeout,concurrency; +static my_string opt_logname=0,opt_update_logname=0, + opt_binlog_index_name = 0,opt_slow_logname=0; +my_string opt_bin_logname = 0; // this one needs to be seen in sql_parse.cc +static char mysql_home[FN_REFLEN],pidfile_name[FN_REFLEN]; +static pthread_t select_thread; +static pthread_t flush_thread; // Used when debugging +static bool opt_log,opt_update_log,opt_bin_log,opt_slow_log,opt_noacl, + opt_disable_networking=0, opt_bootstrap=0,opt_skip_show_db=0, + opt_ansi_mode; +bool opt_sql_bin_update = 0, opt_log_slave_updates = 0; + +// if sql_bin_update is true, SQL_LOG_UPDATE and SQL_LOG_BIN are kept in sync, and are +// treated as aliases for each other + +static bool kill_in_progress=FALSE; +static struct rand_struct sql_rand; +static int cleanup_done; +static char **defaults_argv,time_zone[30]; +static const char *default_table_type_name; + +#ifdef HAVE_OPENSSL +static bool opt_use_ssl = false; +static char *opt_ssl_key = 0; +static char *opt_ssl_cert = 0; +static char *opt_ssl_ca = 0; +static char *opt_ssl_capath = 0; +static VioSSLAcceptorFd* ssl_acceptor_fd = 0; +#endif /* HAVE_OPENSSL */ + +extern bool slave_running; + +I_List<i_string> replicate_do_db, replicate_ignore_db; +// allow the user to tell us which db to replicate and which to ignore +I_List<i_string> binlog_do_db, binlog_ignore_db; + +uint mysql_port; +uint test_flags, select_errors=0, dropping_tables=0,ha_open_options=0; +uint volatile thread_count=0, thread_running=0, kill_cached_threads=0, + wake_thread=0, global_read_lock=0; +ulong thd_startup_options=(OPTION_UPDATE_LOG | OPTION_AUTO_IS_NULL | + OPTION_BIN_LOG | OPTION_AUTO_COMMIT); +uint protocol_version=PROTOCOL_VERSION; +ulong keybuff_size,sortbuff_size,max_item_sort_length,table_cache_size, + max_join_size,join_buff_size,tmp_table_size,thread_stack, + thread_stack_min,net_wait_timeout,what_to_log= ~ (1L << (uint) COM_TIME), + query_buff_size, lower_case_table_names, mysqld_net_retry_count, + net_interactive_timeout, slow_launch_time = 2L, + net_read_timeout,net_write_timeout; +ulong thread_cache_size=0; +volatile ulong cached_thread_count=0; + +// replication parameters, if master_host is not NULL, we are slaving off the master +my_string master_user = (char*) "test", master_password = 0, master_host=0, + master_info_file = (char*) "master.info"; +uint master_port = MYSQL_PORT, master_connect_retry = 60; + +ulong max_tmp_tables,max_heap_table_size; +ulong bytes_sent = 0L, bytes_received = 0L; + +bool opt_endinfo,using_udf_functions,low_priority_updates; +bool volatile abort_loop,select_thread_in_use,flush_thread_in_use,grant_option; +bool volatile ready_to_exit,shutdown_in_progress; +ulong refresh_version=1L,flush_version=1L; /* Increments on each reload */ +ulong query_id=1L,long_query_count,long_query_time,aborted_threads, + aborted_connects,delayed_insert_timeout,delayed_insert_limit, + delayed_queue_size,delayed_insert_threads,delayed_insert_writes, + delayed_rows_in_use,delayed_insert_errors,flush_time; +ulong specialflag=0,opened_tables=0,created_tmp_tables=0; +ulong max_connections,max_insert_delayed_threads,max_used_connections, + max_connect_errors; +ulong thread_id=1L,current_pid; +ulong slow_launch_threads = 0; +char mysql_real_data_home[FN_REFLEN], + mysql_data_home[2],language[LIBLEN],reg_ext[FN_EXTLEN], + default_charset[LIBLEN],mysql_charsets_dir[FN_REFLEN], *charsets_list, + blob_newline,f_fyllchar,max_sort_char,*mysqld_user,*mysqld_chroot, + *opt_init_file; +char server_version[50]=MYSQL_SERVER_VERSION; +const char *first_keyword="first"; +const char **errmesg; /* Error messages */ +byte last_ref[MAX_REFLENGTH]; /* Index ref of keys */ +my_string mysql_unix_port=NULL,mysql_tmpdir=NULL; +ulong my_bind_addr; /* the address we bind to */ +DATE_FORMAT dayord; +double log_10[32]; /* 10 potences */ +I_List<THD> threads,thread_cache; +time_t start_time; + +pthread_key(MEM_ROOT*,THR_MALLOC); +pthread_key(THD*, THR_THD); +pthread_key(NET*, THR_NET); +pthread_mutex_t LOCK_mysql_create_db, LOCK_Acl, LOCK_open, LOCK_thread_count, + LOCK_mapped_file, LOCK_status, LOCK_grant, + LOCK_error_log, + LOCK_delayed_insert, LOCK_delayed_status, LOCK_delayed_create, + LOCK_flush, LOCK_crypt, LOCK_bytes_sent, LOCK_bytes_received, + LOCK_binlog_update, LOCK_slave; + +pthread_cond_t COND_refresh,COND_thread_count,COND_flush, COND_binlog_update, + COND_slave_stopped; +pthread_cond_t COND_thread_cache,COND_flush_thread_cache; +pthread_t signal_thread; +pthread_attr_t connection_attrib; +enum db_type default_table_type=DB_TYPE_MYISAM; + +#ifdef __WIN__ +#undef getpid +#include <process.h> +HANDLE hEventShutdown; +#include "nt_servc.h" +static NTService Service; // Service object for WinNT +#endif + +static void *signal_hand(void *arg); +static void set_options(void); +static void get_options(int argc,char **argv); +static char *get_relative_path(const char *path); +static void fix_paths(void); +static pthread_handler_decl(handle_connections_sockets,arg); +static int bootstrap(FILE *file); +static bool read_init_file(char *file_name); +#ifdef __NT__ +static pthread_handler_decl(handle_connections_namedpipes,arg); +#endif +#ifdef __WIN__ +static int get_service_parameters(); +#endif +static pthread_handler_decl(handle_flush,arg); +extern pthread_handler_decl(handle_slave,arg); +#ifdef SET_RLIMIT_NOFILE +static uint set_maximum_open_files(uint max_file_limit); +#endif + + +/**************************************************************************** +** Code to end mysqld +****************************************************************************/ + +static void close_connections(void) +{ +#ifdef EXTRA_DEBUG + int count=0; +#endif + NET net; + DBUG_ENTER("close_connections"); + + /* Clear thread cache */ + kill_cached_threads++; + flush_thread_cache(); + + /* kill flush thread */ + (void) pthread_mutex_lock(&LOCK_flush); + if (flush_thread_in_use) + { + DBUG_PRINT("quit",("killing flush thread: %lx",flush_thread)); + (void) pthread_cond_signal(&COND_flush); + } + (void) pthread_mutex_unlock(&LOCK_flush); + + /* kill connection thread */ +#if !defined(__WIN__) && !defined(__EMX__) + DBUG_PRINT("quit",("waiting for select thread: %lx",select_thread)); + (void) pthread_mutex_lock(&LOCK_thread_count); + + while (select_thread_in_use) + { + struct timespec abstime; + int error; + LINT_INIT(error); +#ifndef DONT_USE_THR_ALARM + if (pthread_kill(select_thread,THR_CLIENT_ALARM)) + break; // allready dead +#endif +#ifdef HAVE_TIMESPEC_TS_SEC + abstime.ts_sec=time(NULL)+2; // Bsd 2.1 + abstime.ts_nsec=0; +#else + struct timeval tv; + gettimeofday(&tv,0); + abstime.tv_sec=tv.tv_sec+2; + abstime.tv_nsec=tv.tv_usec*1000; +#endif + for (uint tmp=0 ; tmp < 10 ; tmp++) + { + error=pthread_cond_timedwait(&COND_thread_count,&LOCK_thread_count, + &abstime); + if (error != EINTR) + break; + } +#ifdef EXTRA_DEBUG + if (error != 0 && !count++) + sql_print_error("Got error %d from pthread_cond_timedwait",error); +#endif +#if defined(AIX_3_2) || defined(HAVE_DEC_3_2_THREADS) + if (ip_sock != INVALID_SOCKET) + { + VOID(shutdown(ip_sock,2)); + VOID(closesocket(ip_sock)); + VOID(shutdown(unix_sock,2)); + VOID(closesocket(unix_sock)); + VOID(unlink(mysql_unix_port)); + ip_sock=unix_sock= INVALID_SOCKET; + } +#endif + } + (void) pthread_mutex_unlock(&LOCK_thread_count); +#endif /* __WIN__ */ + + + /* Abort listening to new connections */ + DBUG_PRINT("quit",("Closing sockets")); + if ( !opt_disable_networking ) + { + if (ip_sock != INVALID_SOCKET) + { + (void) shutdown(ip_sock,2); + (void) closesocket(ip_sock); + ip_sock= INVALID_SOCKET; + } + } +#ifdef __NT__ + if ( hPipe != INVALID_HANDLE_VALUE ) + { + HANDLE hTempPipe = hPipe; + DBUG_PRINT( "quit", ("Closing named pipes") ); + hPipe = INVALID_HANDLE_VALUE; + CancelIo( hTempPipe ); + DisconnectNamedPipe( hTempPipe ); + CloseHandle( hTempPipe ); + } +#endif +#ifdef HAVE_SYS_UN_H + if (unix_sock != INVALID_SOCKET) + { + (void) shutdown(unix_sock,2); + (void) closesocket(unix_sock); + (void) unlink(mysql_unix_port); + unix_sock= INVALID_SOCKET; + } +#endif + end_thr_alarm(); // Don't allow alarms + + /* First signal all threads that it's time to die */ + + THD *tmp; + (void) pthread_mutex_lock(&LOCK_thread_count); // For unlink from list + + I_List_iterator<THD> it(threads); + while ((tmp=it++)) + { + DBUG_PRINT("quit",("Informing thread %ld that it's time to die", + tmp->thread_id)); + tmp->killed=1; + if (tmp->mysys_var) + { + tmp->mysys_var->abort=1; + if (tmp->mysys_var->current_mutex) + { + pthread_mutex_lock(tmp->mysys_var->current_mutex); + pthread_cond_broadcast(tmp->mysys_var->current_cond); + pthread_mutex_unlock(tmp->mysys_var->current_mutex); + } + } + } + (void) pthread_mutex_unlock(&LOCK_thread_count); // For unlink from list + + if (thread_count) + { + sleep(1); // Give threads time to die + } + + /* Force remaining threads to die by closing the connection to the client */ + + (void) my_net_init(&net, (Vio*) 0); + for (;;) + { + DBUG_PRINT("quit",("Locking LOCK_thread_count")); + (void) pthread_mutex_lock(&LOCK_thread_count); // For unlink from list + if (!(tmp=threads.get())) + { + DBUG_PRINT("quit",("Unlocking LOCK_thread_count")); + (void) pthread_mutex_unlock(&LOCK_thread_count); + break; + } +#ifndef __bsdi__ // Bug in BSDI kernel + if ((net.vio=tmp->net.vio) != 0) + { + sql_print_error(ER(ER_FORCING_CLOSE),my_progname, + tmp->thread_id,tmp->user ? tmp->user : ""); + close_connection(&net,0,0); + } +#endif + DBUG_PRINT("quit",("Unlocking LOCK_thread_count")); + (void) pthread_mutex_unlock(&LOCK_thread_count); + } + net_end(&net); + /* All threads has now been aborted */ + DBUG_PRINT("quit",("Waiting for threads to die (count=%u)",thread_count)); + (void) pthread_mutex_lock(&LOCK_thread_count); + while (thread_count) + { + (void) pthread_cond_wait(&COND_thread_count,&LOCK_thread_count); + DBUG_PRINT("quit",("One thread died (count=%u)",thread_count)); + } + (void) pthread_mutex_unlock(&LOCK_thread_count); + + mysql_log.close(); + mysql_update_log.close(); + mysql_bin_log.close(); + my_free(charsets_list, MYF(0)); + DBUG_PRINT("quit",("close_connections thread")); + DBUG_VOID_RETURN; +} + +void kill_mysql(void) +{ + DBUG_ENTER("kill_mysql"); + +#if defined(__WIN__) + { + if (!SetEvent(hEventShutdown)) + { + DBUG_PRINT("error",("Got error: %ld from SetEvent",GetLastError())); + } + // or: + // HANDLE hEvent=OpenEvent(0, FALSE, "MySqlShutdown"); + // SetEvent(hEventShutdown); + // CloseHandle(hEvent); + } +#elif defined(HAVE_PTHREAD_KILL) + if (pthread_kill(signal_thread,SIGTERM)) /* End everything nicely */ + { + DBUG_PRINT("error",("Got error %d from pthread_kill",errno)); /* purecov: inspected */ + } +#else + kill(current_pid,SIGTERM); +#endif + DBUG_PRINT("quit",("After pthread_kill")); + shutdown_in_progress=1; // Safety if kill didn't work + DBUG_VOID_RETURN; +} + + + /* Force server down. kill all connections and threads and exit */ + +#ifndef __WIN__ +static void *kill_server(void *sig_ptr) +#define RETURN_FROM_KILL_SERVER return 0 +#else +static void __cdecl kill_server(int sig_ptr) +#define RETURN_FROM_KILL_SERVER return +#endif +{ + int sig=(int) (long) sig_ptr; // This is passed a int + DBUG_ENTER("kill_server"); + + // if there is a signal during the kill in progress, we do not need + // another one + if (kill_in_progress) // Safety + RETURN_FROM_KILL_SERVER; + kill_in_progress=TRUE; + abort_loop=1; // This should be set + signal(sig,SIG_IGN); + if (sig == MYSQL_KILL_SIGNAL || sig == 0) + sql_print_error(ER(ER_NORMAL_SHUTDOWN),my_progname); + else + sql_print_error(ER(ER_GOT_SIGNAL),my_progname,sig); /* purecov: inspected */ + +#if defined(USE_ONE_SIGNAL_HAND) && !defined(__WIN__) + my_thread_init(); // If this is a new thread +#endif + close_connections(); + sql_print_error(ER(ER_SHUTDOWN_COMPLETE),my_progname); + if (sig != MYSQL_KILL_SIGNAL && sig != 0) + unireg_abort(1); /* purecov: inspected */ + else + unireg_end(0); + pthread_exit(0); /* purecov: deadcode */ + RETURN_FROM_KILL_SERVER; +} + + +#ifdef USE_ONE_SIGNAL_HAND +pthread_handler_decl(kill_server_thread,arg __attribute__((unused))) +{ + my_thread_init(); // Initialize new thread + kill_server(0); + my_thread_end(); // Normally never reached + return 0; +} +#endif + +static sig_handler print_signal_warning(int sig) +{ + sql_print_error("Warning: Got signal %d from thread %d", + sig,my_thread_id()); +#ifdef DONT_REMEMBER_SIGNAL + sigset(sig,print_signal_warning); /* int. thread system calls */ +#endif +#ifndef __WIN__ + if (sig == SIGALRM) + alarm(2); /* reschedule alarm */ +#endif +} + + +void unireg_end(int signal_number __attribute__((unused))) +{ + clean_up(); + pthread_exit(0); // Exit is in main thread +} + + +void unireg_abort(int exit_code) +{ + if (exit_code) + sql_print_error("Aborting\n"); + (void) my_delete(pidfile_name,MYF(0)); // This may not always exist + clean_up(); /* purecov: inspected */ + exit(exit_code); /* purecov: inspected */ +} + + +void clean_up(void) +{ + DBUG_PRINT("exit",("clean_up")); + if (cleanup_done++) + return; /* purecov: inspected */ + acl_free(1); + grant_free(); + sql_cache_free(); + table_cache_free(); + hostname_cache_free(); + item_user_lock_free(); + lex_free(); /* Free some memory */ +#ifdef HAVE_DLOPEN + if (!opt_noacl) + udf_free(); +#endif + end_key_cache(); /* This is usually freed automaticly */ + (void) ha_panic(HA_PANIC_CLOSE); /* close all tables */ +#ifdef USE_RAID + end_raid(); +#endif + x_free((gptr) errmsg[ERRMAPP]); /* Free messages */ + free_defaults(defaults_argv); + my_free(mysql_tmpdir,MYF(0)); + my_end(opt_endinfo ? MY_CHECK_ERROR | MY_GIVE_INFO : 0); + + /* Tell main we are ready */ + (void) pthread_mutex_lock(&LOCK_thread_count); + ready_to_exit=1; + (void) pthread_cond_broadcast(&COND_thread_count); + (void) pthread_mutex_unlock(&LOCK_thread_count); +} /* clean_up */ + + + +/**************************************************************************** +** Init IP and UNIX socket +****************************************************************************/ + +static void set_ports() +{ + char *env; + if (!mysql_port) + { // Get port if not from commandline + struct servent *serv_ptr; + mysql_port = MYSQL_PORT; + if ((serv_ptr = getservbyname("mysql", "tcp"))) + mysql_port = ntohs((u_short) serv_ptr->s_port); /* purecov: inspected */ + if ((env = getenv("MYSQL_TCP_PORT"))) + mysql_port = (uint) atoi(env); /* purecov: inspected */ + } + if (!mysql_unix_port) + { +#ifdef __WIN__ + mysql_unix_port = (char*) MYSQL_NAMEDPIPE; +#else + mysql_unix_port = (char*) MYSQL_UNIX_ADDR; +#endif + if ((env = getenv("MYSQL_UNIX_PORT"))) + mysql_unix_port = env; /* purecov: inspected */ + } +} + +/* Change to run as another user if started with --user */ + +static void set_user(const char *user) +{ +#ifndef __WIN__ + struct passwd *ent; + + // don't bother if we aren't superuser + if (geteuid()) + { + if (user) + fprintf(stderr, + "Warning: One can only use the --user switch if running as root\n"); + return; + } + else if (!user) + { + if (!opt_bootstrap) + { + fprintf(stderr,"Fatal error: Please read \"Security\" section of the manual to find out how to run mysqld as root!\n"); + unireg_abort(1); + } + return; + } + if (!strcmp(user,"root")) + return; // Avoid problem with dynamic libraries + + if (!(ent = getpwnam(user))) + { + fprintf(stderr,"Fatal error: Can't change to run as user '%s' ; Please check that the user exists!\n",user); + unireg_abort(1); + } +#ifdef HAVE_INITGROUPS + initgroups(user,ent->pw_gid); +#endif + if (setgid(ent->pw_gid) == -1) + { + sql_perror("setgid"); + unireg_abort(1); + } + if (setuid(ent->pw_uid) == -1) + { + sql_perror("setuid"); + unireg_abort(1); + } +#endif +} + +/* Change root user if started with --chroot */ + +static void set_root(const char *path) +{ +#if !defined(__WIN__) && !defined(__EMX__) + if (chroot(path) == -1) + { + sql_perror("chroot"); + unireg_abort(1); + } +#endif +} + +static void server_init(void) +{ + struct sockaddr_in IPaddr; +#ifdef HAVE_SYS_UN_H + struct sockaddr_un UNIXaddr; +#endif + int arg=1; + DBUG_ENTER("server_init"); + +#ifdef __WIN__ + if ( !opt_disable_networking ) + { + WSADATA WsaData; + if (SOCKET_ERROR == WSAStartup (0x0101, &WsaData)) + { + my_message(0,"WSAStartup Failed\n",MYF(0)); + unireg_abort(1); + } + } +#endif /* __WIN__ */ + + set_ports(); + + if (mysql_port != 0 && !opt_disable_networking && !opt_bootstrap) + { + DBUG_PRINT("general",("IP Socket is %d",mysql_port)); + ip_sock = socket(AF_INET, SOCK_STREAM, 0); + if (ip_sock == INVALID_SOCKET) + { + DBUG_PRINT("error",("Got error: %d from socket()",socket_errno)); + sql_perror(ER(ER_IPSOCK_ERROR)); /* purecov: tested */ + unireg_abort(1); /* purecov: tested */ + } + bzero((char*) &IPaddr, sizeof(IPaddr)); + IPaddr.sin_family = AF_INET; + IPaddr.sin_addr.s_addr = my_bind_addr; + IPaddr.sin_port = (unsigned short) htons((unsigned short) mysql_port); + (void) setsockopt(ip_sock,SOL_SOCKET,SO_REUSEADDR,(char*)&arg,sizeof(arg)); + for(;;) + { + if (bind(ip_sock, my_reinterpret_cast(struct sockaddr *) (&IPaddr), + sizeof(IPaddr)) >= 0) + break; + DBUG_PRINT("error",("Got error: %d from bind",socket_errno)); + sql_perror("Can't start server: Bind on TCP/IP port");/* Had a loop here */ + sql_print_error("Do you already have another mysqld server running on port: %d ?",mysql_port); + unireg_abort(1); + } + (void) listen(ip_sock,(int) back_log); + } + + if (mysqld_chroot) + set_root(mysqld_chroot); + + set_user(mysqld_user); // set_user now takes care of mysqld_user==NULL + +#ifdef __NT__ + /* create named pipe */ + if (Service.IsNT() && mysql_unix_port[0] && !opt_bootstrap) + { + sprintf( szPipeName, "\\\\.\\pipe\\%s", mysql_unix_port ); + ZeroMemory( &saPipeSecurity, sizeof(saPipeSecurity) ); + ZeroMemory( &sdPipeDescriptor, sizeof(sdPipeDescriptor) ); + if ( !InitializeSecurityDescriptor(&sdPipeDescriptor, + SECURITY_DESCRIPTOR_REVISION) ) + { + sql_perror("Can't start server : Initialize security descriptor"); + unireg_abort(1); + } + if (!SetSecurityDescriptorDacl(&sdPipeDescriptor, TRUE, NULL, FALSE)) + { + sql_perror("Can't start server : Set security descriptor"); + unireg_abort(1); + } + saPipeSecurity.nLength = sizeof( SECURITY_ATTRIBUTES ); + saPipeSecurity.lpSecurityDescriptor = &sdPipeDescriptor; + saPipeSecurity.bInheritHandle = FALSE; + if ((hPipe = CreateNamedPipe(szPipeName, + PIPE_ACCESS_DUPLEX, + PIPE_TYPE_BYTE | + PIPE_READMODE_BYTE | + PIPE_WAIT, + PIPE_UNLIMITED_INSTANCES, + (int) net_buffer_length, + (int) net_buffer_length, + NMPWAIT_USE_DEFAULT_WAIT, + &saPipeSecurity )) == INVALID_HANDLE_VALUE) + { + LPVOID lpMsgBuf; + int error=GetLastError(); + FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | + FORMAT_MESSAGE_FROM_SYSTEM, + NULL, error, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), + (LPTSTR) &lpMsgBuf, 0, NULL ); + MessageBox( NULL, (LPTSTR) lpMsgBuf, "Error from CreateNamedPipe", + MB_OK|MB_ICONINFORMATION ); + LocalFree( lpMsgBuf ); + unireg_abort(1); + } + } +#endif + +#if defined(HAVE_SYS_UN_H) && !defined(HAVE_mit_thread) + /* + ** Create the UNIX socket + */ + if (mysql_unix_port[0] && !opt_bootstrap) + { + DBUG_PRINT("general",("UNIX Socket is %s",mysql_unix_port)); + + if ((unix_sock = socket(AF_UNIX, SOCK_STREAM, 0)) < 0) + { + sql_perror("Can't start server : UNIX Socket "); /* purecov: inspected */ + unireg_abort(1); /* purecov: inspected */ + } + bzero((char*) &UNIXaddr, sizeof(UNIXaddr)); + UNIXaddr.sun_family = AF_UNIX; + strmov(UNIXaddr.sun_path, mysql_unix_port); + (void) unlink(mysql_unix_port); + (void) setsockopt(unix_sock,SOL_SOCKET,SO_REUSEADDR,(char*)&arg, + sizeof(arg)); + umask(0); + if (bind(unix_sock, my_reinterpret_cast(struct sockaddr *) (&UNIXaddr), + sizeof(UNIXaddr)) < 0) + { + sql_perror("Can't start server : Bind on unix socket"); /* purecov: tested */ + sql_print_error("Do you already have another mysqld server running on socket: %s ?",mysql_unix_port); + unireg_abort(1); /* purecov: tested */ + } + umask(((~my_umask) & 0666)); +#if defined(S_IFSOCK) && defined(SECURE_SOCKETS) + (void) chmod(mysql_unix_port,S_IFSOCK); /* Fix solaris 2.6 bug */ +#endif + (void) listen(unix_sock,(int) back_log); + } +#endif + DBUG_PRINT("info",("server started")); + DBUG_VOID_RETURN; +} + + +void yyerror(const char *s) +{ + NET *net=my_pthread_getspecific_ptr(NET*,THR_NET); + char *yytext=(char*) current_lex->tok_start; + if (!strcmp(s,"parse error")) + s=ER(ER_SYNTAX_ERROR); + net_printf(net,ER_PARSE_ERROR, s, yytext ? (char*) yytext : "", + current_lex->yylineno); +} + + +void close_connection(NET *net,uint errcode,bool lock) +{ + Vio* vio; + DBUG_ENTER("close_connection"); + DBUG_PRINT("enter",("fd: %s error: '%s'", + net->vio? vio_description(net->vio):"(not connected)", + errcode ? ER(errcode) : "")); + if (lock) + (void) pthread_mutex_lock(&LOCK_thread_count); + if ((vio=net->vio) != 0) + { + if (errcode) + send_error(net,errcode,ER(errcode)); /* purecov: inspected */ + vio_close(vio); /* vio is freed in delete thd */ + } + if (lock) + (void) pthread_mutex_unlock(&LOCK_thread_count); + DBUG_VOID_RETURN; +} + + /* Called when a thread is aborted */ + /* ARGSUSED */ + +sig_handler end_thread_signal(int sig __attribute__((unused))) +{ + THD *thd=current_thd; + DBUG_ENTER("end_thread_signal"); + if (thd) + end_thread(thd,0); + DBUG_VOID_RETURN; /* purecov: deadcode */ +} + + +void end_thread(THD *thd, bool put_in_cache) +{ + DBUG_ENTER("end_thread"); + (void) pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + delete thd; + + if (cached_thread_count < thread_cache_size && ! abort_loop && + !kill_cached_threads) + { + /* Don't kill the thread, just put it in cache for reuse */ + DBUG_PRINT("info", ("Adding thread to cache")) + cached_thread_count++; + while (!abort_loop && ! wake_thread && ! kill_cached_threads) + (void) pthread_cond_wait(&COND_thread_cache, &LOCK_thread_count); + cached_thread_count--; + if (kill_cached_threads) + pthread_cond_signal(&COND_flush_thread_cache); + if (wake_thread) + { + wake_thread--; + thd=thread_cache.get(); + threads.append(thd); + (void) thd->store_globals(); + pthread_mutex_unlock(&LOCK_thread_count); + DBUG_VOID_RETURN; + } + } + + DBUG_PRINT("info", ("sending a broadcast")) + + /* Tell main we are ready */ + (void) pthread_cond_broadcast(&COND_thread_count); + (void) pthread_mutex_unlock(&LOCK_thread_count); + DBUG_PRINT("info", ("unlocked thread_count mutex")) +#ifndef DBUG_OFF + if (!(test_flags & TEST_NO_THREADS)) // For debugging under Linux +#endif + { + my_thread_end(); + pthread_exit(0); + } + DBUG_VOID_RETURN; +} + + +/* Start a cached thread. LOCK_thread_count is locked on entry */ + +static void start_cached_thread(THD *thd) +{ + thread_cache.append(thd); + wake_thread++; + thread_count++; + pthread_cond_signal(&COND_thread_cache); +} + + +void flush_thread_cache() +{ + (void) pthread_mutex_lock(&LOCK_thread_count); + kill_cached_threads++; + while (cached_thread_count) + { + pthread_cond_broadcast(&COND_thread_cache); + pthread_cond_wait(&COND_flush_thread_cache,&LOCK_thread_count); + } + kill_cached_threads--; + (void) pthread_mutex_unlock(&LOCK_thread_count); +} + + + /* + ** Aborts a thread nicely. Commes here on SIGPIPE + ** TODO: One should have to fix that thr_alarm know about this + ** thread too + */ + +#ifdef THREAD_SPECIFIC_SIGPIPE +static sig_handler abort_thread(int sig __attribute__((unused))) +{ + THD *thd=current_thd; + DBUG_ENTER("abort_thread"); + if (thd) + thd->killed=1; + DBUG_VOID_RETURN; +} +#endif + +/****************************************************************************** +** Setup a signal thread with handles all signals +** Because linux doesn't support scemas use a mutex to check that +** the signal thread is ready before continuing +******************************************************************************/ + +#ifdef __WIN__ +static void init_signals(void) +{ + int signals[] = {SIGINT,SIGILL,SIGFPE,SIGSEGV,SIGTERM,SIGABRT } ; + for (uint i=0 ; i < sizeof(signals)/sizeof(int) ; i++) + signal( signals[i], kill_server) ; + signal(SIGBREAK,SIG_IGN); //ignore SIGBREAK for NT +} + +#elif defined(__EMX__) +static void sig_reload(int signo) +{ + reload_acl_and_cache(~0); // Flush everything + signal(signo, SIG_ACK); +} + +static void sig_kill(int signo) +{ + if (!abort_loop) + { + abort_loop=1; // mark abort for threads + kill_server((void*) signo); + } + signal(signo, SIG_ACK); +} + +static void init_signals(void) +{ + signal(SIGQUIT, sig_kill); + signal(SIGKILL, sig_kill); + signal(SIGTERM, sig_kill); + signal(SIGINT, sig_kill); + signal(SIGHUP, sig_reload); // Flush everything + signal(SIGALRM, SIG_IGN); + signal(SIGBREAK,SIG_IGN); + signal_thread = pthread_self(); +} +#else + +static void init_signals(void) +{ + sigset_t set; + pthread_attr_t thr_attr; + int error; + DBUG_ENTER("init_signals"); + + sigset(THR_KILL_SIGNAL,end_thread_signal); + sigset(THR_SERVER_ALARM,print_signal_warning); // Should never be called! + (void) sigemptyset(&set); +#ifdef THREAD_SPECIFIC_SIGPIPE + sigset(SIGPIPE,abort_thread); + sigaddset(&set,SIGPIPE); +#else + (void) signal(SIGPIPE,SIG_IGN); // Can't know which thread + sigaddset(&set,SIGPIPE); +#endif + sigaddset(&set,SIGINT); + sigaddset(&set,SIGQUIT); + sigaddset(&set,SIGTERM); + sigaddset(&set,SIGHUP); + signal(SIGTERM,SIG_DFL); // If it's blocked by parent +#ifdef SIGTSTP + sigaddset(&set,SIGTSTP); +#endif + sigaddset(&set,THR_SERVER_ALARM); + sigdelset(&set,THR_KILL_SIGNAL); // May be SIGINT + sigdelset(&set,THR_CLIENT_ALARM); // For alarms + (void) pthread_sigmask(SIG_SETMASK,&set,NULL); + + (void) pthread_attr_init(&thr_attr); +#if !defined(HAVE_DEC_3_2_THREADS) + pthread_attr_setscope(&thr_attr,PTHREAD_SCOPE_SYSTEM); + (void) pthread_attr_setdetachstate(&thr_attr,PTHREAD_CREATE_DETACHED); + if (!(opt_specialflag & SPECIAL_NO_PRIOR)) + my_pthread_attr_setprio(&thr_attr,INTERRUPT_PRIOR); + pthread_attr_setstacksize(&thr_attr,32768); +#endif + + (void) pthread_mutex_lock(&LOCK_thread_count); + if ((error=pthread_create(&signal_thread,&thr_attr,signal_hand,0))) + { + sql_print_error("Can't create interrupt-thread (error %d, errno: %d)", + error,errno); + exit(1); + } + (void) pthread_cond_wait(&COND_thread_count,&LOCK_thread_count); + pthread_mutex_unlock(&LOCK_thread_count); + + (void) pthread_attr_destroy(&thr_attr); + DBUG_VOID_RETURN; +} + + +/* +** This threads handles all signals and alarms +*/ + +/* ARGSUSED */ +static void *signal_hand(void *arg __attribute__((unused))) +{ + sigset_t set; + int sig; + my_thread_init(); // Init new thread + DBUG_ENTER("signal_hand"); + + /* Setup alarm handler */ + init_thr_alarm(max_connections+max_insert_delayed_threads); +#if SIGINT != THR_KILL_SIGNAL + (void) sigemptyset(&set); // Setup up SIGINT for debug + (void) sigaddset(&set,SIGINT); // For debugging + (void) pthread_sigmask(SIG_UNBLOCK,&set,NULL); +#endif + (void) sigemptyset(&set); // Setup up SIGINT for debug +#ifdef USE_ONE_SIGNAL_HAND + (void) sigaddset(&set,THR_SERVER_ALARM); // For alarms +#endif + (void) sigaddset(&set,SIGQUIT); + (void) sigaddset(&set,SIGTERM); +#if THR_CLIENT_ALARM != SIGHUP + (void) sigaddset(&set,SIGHUP); +#endif + (void) sigaddset(&set,SIGTSTP); + + /* Save pid to this process (or thread on Linux) */ + { + FILE *pidFile; + if ((pidFile = my_fopen(pidfile_name,O_WRONLY,MYF(MY_WME)))) + { + fprintf(pidFile,"%lu",(ulong) getpid()); + (void) my_fclose(pidFile,MYF(0)); + (void) chmod(pidfile_name,0644); + } + } + + (void) pthread_mutex_lock(&LOCK_thread_count); + (void) pthread_cond_signal(&COND_thread_count); /* continue init_signals */ + (void) pthread_mutex_unlock(&LOCK_thread_count); + + for (;;) + { + int error; // Used when debugging + if (shutdown_in_progress && !abort_loop) + { + sig=SIGTERM; + error=0; + } + else + while ((error=my_sigwait(&set,&sig)) == EINTR) ; + if (cleanup_done) + pthread_exit(0); // Safety + switch (sig) { + case SIGTERM: + case SIGQUIT: + case SIGKILL: +#ifdef EXTRA_DEBUG + sql_print_error("Got signal %d to shutdown mysqld",sig); +#endif + DBUG_PRINT("info",("Got signal: %d abort_loop: %d",sig,abort_loop)); + if (!abort_loop) + { + abort_loop=1; // mark abort for threads +#ifdef USE_ONE_SIGNAL_HAND + pthread_t tmp; + if (!(opt_specialflag & SPECIAL_NO_PRIOR)) + my_pthread_attr_setprio(&connection_attrib,INTERRUPT_PRIOR); + if (pthread_create(&tmp,&connection_attrib, kill_server_thread, + (void*) sig)) + sql_print_error("Error: Can't create thread to kill server"); +#else + kill_server((void*) sig); // MIT THREAD has a alarm thread +#endif + } + break; + case SIGHUP: + reload_acl_and_cache(~0); // Flush everything + mysql_print_status((THD*) 0); // Send debug some info + break; +#ifdef USE_ONE_SIGNAL_HAND + case THR_SERVER_ALARM: + process_alarm(sig); // Trigger alarms. + break; +#endif + default: +#ifdef EXTRA_DEBUG + sql_print_error("Warning: Got signal: %d, error: %d",sig,error); /* purecov: tested */ +#endif + break; /* purecov: tested */ + } + } + return(0); /* purecov: deadcode */ +} + +#endif /* __WIN__*/ + + +/* +** All global error messages are sent here where the first one is stored for +** the client +*/ + + +/* ARGSUSED */ +static int my_message_sql(uint error, const char *str, + myf MyFlags __attribute__((unused))) +{ + NET *net; + DBUG_ENTER("my_message_sql"); + DBUG_PRINT("error",("Message: '%s'",str)); + if ((net=my_pthread_getspecific_ptr(NET*,THR_NET))) + { + if (!net->last_error[0]) // Return only first message + { + strmake(net->last_error,str,sizeof(net->last_error)-1); + net->last_errno=error ? error : ER_UNKNOWN_ERROR; + } + } + else + sql_print_error("%s: %s",my_progname,str); /* purecov: inspected */ + DBUG_RETURN(0); +} + +#ifdef __WIN__ +#undef errno +#undef EINTR +#define errno WSAGetLastError() +#define EINTR WSAEINTR + +struct utsname +{ + char nodename[FN_REFLEN]; +}; + +int uname(struct utsname *a) +{ + return -1; +} +#endif + + +#ifdef __WIN__ +pthread_handler_decl(handle_shutdown,arg) +{ + MSG msg; + my_thread_init(); + + /* this call should create the message queue for this thread */ + PeekMessage(&msg, NULL, 1, 65534,PM_NOREMOVE); + + if (WaitForSingleObject(hEventShutdown,INFINITE)==WAIT_OBJECT_0) + kill_server(MYSQL_KILL_SIGNAL); + return 0; +} + +int __stdcall handle_kill(ulong ctrl_type) +{ + if (ctrl_type == CTRL_CLOSE_EVENT || + ctrl_type == CTRL_SHUTDOWN_EVENT) + { + kill_server(MYSQL_KILL_SIGNAL); + return TRUE; + } + return FALSE; +} +#endif + +const char *load_default_groups[]= { "mysqld","server",0 }; + +#ifdef HAVE_LIBWRAP +char *libwrapName=NULL; +#endif + +static void open_log(MYSQL_LOG *log, const char *hostname, + const char *opt_name, const char *extension, + enum_log_type type) +{ + char tmp[FN_REFLEN]; + if (!opt_name || !opt_name[0]) + { + strnmov(tmp,hostname,FN_REFLEN-5); + strmov(strcend(tmp,'.'),extension); + opt_name=tmp; + } + log->open(opt_name,type); +} + + + +#ifdef __WIN__ +int win_main(int argc, char **argv) +#else +int main(int argc, char **argv) +#endif +{ + DEBUGGER_OFF; + char hostname[FN_REFLEN]; + + my_umask=0660; // Default umask for new files + my_umask_dir=0700; // Default umask for new directories + MY_INIT(argv[0]); // init my_sys library & pthreads + tzset(); // Set tzname + + start_time=time((time_t*) 0); +#ifdef HAVE_TZNAME +#if defined(HAVE_LOCALTIME_R) && defined(_REENTRANT) + { + struct tm tm_tmp; + localtime_r(&start_time,&tm_tmp); + strmov(time_zone,tzname[tm_tmp.tm_isdst == 1 ? 1 : 0]); + } +#else + { + struct tm *start_tm; + start_tm=localtime(&start_time); + strmov(time_zone=tzname[start_tm->tm_isdst == 1 ? 1 : 0]); + } +#endif +#endif + + if (gethostname(hostname,sizeof(hostname)-4) < 0) + strmov(hostname,"mysql"); + strmov(pidfile_name,hostname); + strmov(strcend(pidfile_name,'.'),".pid"); // Add extension +#ifdef DEMO_VERSION + strcat(server_version,"-demo"); +#endif +#ifdef SHAREWARE_VERSION + strcat(server_version,"-shareware"); +#endif +#ifndef DBUG_OFF + strcat(server_version,"-debug"); +#endif +#ifdef _CUSTOMSTARTUPCONFIG_ + if (_cust_check_startup()) + { + /* _cust_check_startup will report startup failure error */ + exit( 1 ); + } +#endif + load_defaults("my",load_default_groups,&argc,&argv); + defaults_argv=argv; + mysql_tmpdir=getenv("TMPDIR"); /* Use this if possible */ +#ifdef __WIN__ + if (!mysql_tmpdir) + mysql_tmpdir=getenv("TEMP"); + if (!mysql_tmpdir) + mysql_tmpdir=getenv("TMP"); +#endif + if (!mysql_tmpdir || !mysql_tmpdir[0]) + mysql_tmpdir=(char*) P_tmpdir; /* purecov: inspected */ + + set_options(); +#ifdef __WIN__ + /* service parameters can be overwritten by options */ + if (get_service_parameters()) + { + my_message( 0, "Can't read MySQL service parameters", MYF(0) ); + exit( 1 ); + } +#endif + get_options(argc,argv); + if (opt_log || opt_update_log || opt_slow_log || opt_bin_log) + strcat(server_version,"-log"); + DBUG_PRINT("info",("%s Ver %s for %s on %s\n",my_progname, + server_version, SYSTEM_TYPE,MACHINE_TYPE)); + + /* These must be set early */ + + (void) pthread_cond_init(&COND_thread_count,NULL); + (void) pthread_mutex_init(&LOCK_mysql_create_db,NULL); + (void) pthread_mutex_init(&LOCK_Acl,NULL); + (void) pthread_mutex_init(&LOCK_grant,NULL); + (void) pthread_mutex_init(&LOCK_open,NULL); + (void) pthread_mutex_init(&LOCK_thread_count,NULL); + (void) pthread_mutex_init(&LOCK_mapped_file,NULL); + (void) pthread_mutex_init(&LOCK_status,NULL); + (void) pthread_mutex_init(&LOCK_error_log,NULL); + (void) pthread_mutex_init(&LOCK_delayed_insert,NULL); + (void) pthread_mutex_init(&LOCK_delayed_status,NULL); + (void) pthread_mutex_init(&LOCK_delayed_create,NULL); + (void) pthread_cond_init(&COND_refresh,NULL); + (void) pthread_cond_init(&COND_thread_cache,NULL); + (void) pthread_cond_init(&COND_flush_thread_cache,NULL); + (void) pthread_cond_init(&COND_flush,NULL); + (void) pthread_mutex_init(&LOCK_flush,NULL); + (void) pthread_mutex_init(&LOCK_crypt,NULL); + (void) pthread_mutex_init(&LOCK_bytes_sent,NULL); + (void) pthread_mutex_init(&LOCK_bytes_received,NULL); + (void) pthread_mutex_init(&LOCK_timezone,NULL); + (void) pthread_mutex_init(&LOCK_binlog_update, NULL); + (void) pthread_mutex_init(&LOCK_slave, NULL); + (void) pthread_cond_init(&COND_binlog_update, NULL); + (void) pthread_cond_init(&COND_slave_stopped, NULL); + + if (set_default_charset_by_name(default_charset, MYF(MY_WME))) + unireg_abort(1); + charsets_list = list_charsets(MYF(MY_COMPILED_SETS|MY_CONFIG_SETS)); + +#ifdef HAVE_OPENSSL + if (opt_use_ssl) + { + ssl_acceptor_fd = VioSSLAcceptorFd_new(opt_ssl_key, opt_ssl_cert, + opt_ssl_ca, opt_ssl_capath); + if (!ssl_acceptor_fd) + opt_use_ssl=0; + /* having ssl_acceptor_fd!=0 signals the use of SSL */ + } +#endif /* HAVE_OPENSSL */ + +#ifdef HAVE_LIBWRAP + libwrapName= my_progname+dirname_length(my_progname); + openlog(libwrapName, LOG_PID, LOG_AUTH); +#endif + + if (!(opt_specialflag & SPECIAL_NO_PRIOR)) + my_pthread_setprio(pthread_self(),CONNECT_PRIOR); + /* Parameter for threads created for connections */ + (void) pthread_attr_init(&connection_attrib); + (void) pthread_attr_setdetachstate(&connection_attrib, + PTHREAD_CREATE_DETACHED); + pthread_attr_setstacksize(&connection_attrib,thread_stack); + + if (!(opt_specialflag & SPECIAL_NO_PRIOR)) + my_pthread_attr_setprio(&connection_attrib,WAIT_PRIOR); + pthread_attr_setscope(&connection_attrib, PTHREAD_SCOPE_SYSTEM); + +#ifdef SET_RLIMIT_NOFILE + /* connections and databases neads lots of files */ + { + uint wanted_files=10+(uint) max(max_connections*5, + max_connections+table_cache_size*2); + uint files=set_maximum_open_files(wanted_files); + if (files && files < wanted_files) // Some systems return 0 + { + max_connections= (ulong) (files-10)/5; + table_cache_size= (ulong) (files-10-max_connections)/2; + DBUG_PRINT("warning", + ("Changed limits: max_connections: %ld table_cache: %ld", + max_connections,table_cache_size)); + sql_print_error("Warning: Changed limits: max_connections: %ld table_cache: %ld",max_connections,table_cache_size); + } + } +#endif + unireg_init(opt_specialflag); /* Set up extern variabels */ + init_errmessage(); /* Read error messages from file */ + lex_init(); + item_init(); + mysys_uses_curses=0; +#ifdef USE_REGEX + regex_init(); +#endif + select_thread=pthread_self(); + select_thread_in_use=1; + + /* + ** We have enough space for fiddling with the argv, continue + */ + umask(((~my_umask) & 0666)); + if (my_setwd(mysql_real_data_home,MYF(MY_WME))) + { + unireg_abort(1); /* purecov: inspected */ + } + mysql_data_home[0]=FN_CURLIB; // all paths are relative from here + mysql_data_home[1]=0; + server_init(); + table_cache_init(); + hostname_cache_init(); + sql_cache_init(); + randominit(&sql_rand,(ulong) start_time,(ulong) start_time/2); + reset_floating_point_exceptions(); + init_thr_lock(); + + /* Setup log files */ + if (opt_log) + open_log(&mysql_log, hostname, opt_logname, ".log", LOG_NORMAL); + if (opt_update_log) + open_log(&mysql_update_log, hostname, opt_update_logname, "", + LOG_NEW); + if (opt_bin_log) + { + mysql_bin_log.set_index_file_name(opt_binlog_index_name); + open_log(&mysql_bin_log, hostname, opt_bin_logname, "-bin", + LOG_BIN); + } + if (opt_slow_log) + open_log(&mysql_slow_log, hostname, opt_slow_logname, "-slow.log", + LOG_NORMAL); + if (ha_init()) + { + sql_print_error("Can't init databases"); + exit(1); + } + ft_init_stopwords(NULL); /* SerG */ + +#ifdef __WIN__ +#define MYSQL_ERR_FILE "mysql.err" + if (!opt_console) + { + freopen(MYSQL_ERR_FILE,"a+",stdout); + freopen(MYSQL_ERR_FILE,"a+",stderr); + FreeConsole(); // Remove window + } +#endif + + /* + init signals & alarm + After this we can't quit by a simple unireg_abort + */ + error_handler_hook = my_message_sql; + if (pthread_key_create(&THR_THD,NULL) || pthread_key_create(&THR_NET,NULL) || + pthread_key_create(&THR_MALLOC,NULL)) + { + sql_print_error("Can't create thread-keys"); + exit(1); + } + init_signals(); // Creates pidfile + if (acl_init(opt_noacl)) + { + select_thread_in_use=0; + (void) pthread_kill(signal_thread,MYSQL_KILL_SIGNAL); + (void) my_delete(pidfile_name,MYF(MY_WME)); // Not neaded anymore + exit(1); + } + if (!opt_noacl) + (void) grant_init(); + +#ifdef HAVE_DLOPEN + if (!opt_noacl) + udf_init(); +#endif + + if (opt_bootstrap) + { + int error=bootstrap(stdin); + end_thr_alarm(); // Don't allow alarms + unireg_abort(error ? 1 : 0); + } + if (opt_init_file) + { + if (read_init_file(opt_init_file)) + { + end_thr_alarm(); // Don't allow alarms + unireg_abort(1); + } + } + (void) thr_setconcurrency(concurrency); // 10 by default +#ifdef __WIN__ //IRENA + { + hEventShutdown=CreateEvent(0, FALSE, FALSE, "MySqlShutdown"); + pthread_t hThread; + if (pthread_create(&hThread,&connection_attrib,handle_shutdown,0)) + sql_print_error("Warning: Can't create thread to handle shutdown requests"); + + // On "Stop Service" we have to do regular shutdown + Service.SetShutdownEvent(hEventShutdown); + } +#endif + + if (flush_time && flush_time != ~(ulong) 0L) + { + pthread_t hThread; + if (pthread_create(&hThread,&connection_attrib,handle_flush,0)) + sql_print_error("Warning: Can't create thread to handle flush"); + } + + // slave thread + if(master_host) + { + pthread_t hThread; + if(pthread_create(&hThread, &connection_attrib, handle_slave, 0)) + sql_print_error("Warning: Can't create thread to handle slave"); + + } + + printf(ER(ER_READY),my_progname,server_version,""); + fflush(stdout); + +#ifdef __NT__ + if (hPipe == INVALID_HANDLE_VALUE && !have_tcpip) + { + sql_print_error("TCP/IP must be installed on Win98 platforms"); + } + else + { + pthread_mutex_lock(&LOCK_thread_count); + (void) pthread_cond_init(&COND_handler_count,NULL); + { + pthread_t hThread; + handler_count=0; + if ( hPipe != INVALID_HANDLE_VALUE ) + { + handler_count++; + if (pthread_create(&hThread,&connection_attrib, + handle_connections_namedpipes, 0)) + { + sql_print_error("Warning: Can't create thread to handle named pipes"); + handler_count--; + } + } + if (have_tcpip) + { + handler_count++; + if (pthread_create(&hThread,&connection_attrib, + handle_connections_sockets, 0)) + { + sql_print_error("Warning: Can't create thread to handle named pipes"); + handler_count--; + } + } + while (handler_count > 0) + { + pthread_cond_wait(&COND_handler_count,&LOCK_thread_count); + } + } + pthread_mutex_unlock(&LOCK_thread_count); + } +#else + handle_connections_sockets(0); +#ifdef EXTRA_DEBUG + sql_print_error("Exiting main thread"); +#endif +#endif /* __NT__ */ + + /* (void) pthread_attr_destroy(&connection_attrib); */ + + DBUG_PRINT("quit",("Exiting main thread")); + +#ifndef __WIN__ +#ifdef EXTRA_DEBUG + sql_print_error("Before Lock_thread_count"); +#endif + (void) pthread_mutex_lock(&LOCK_thread_count); + select_thread_in_use=0; // For close_connections + (void) pthread_cond_broadcast(&COND_thread_count); + (void) pthread_mutex_unlock(&LOCK_thread_count); +#ifdef EXTRA_DEBUG + sql_print_error("After lock_thread_count"); +#endif +#else + // remove the event, because it will not be valid anymore + Service.SetShutdownEvent(0); + if(hEventShutdown) CloseHandle(hEventShutdown); + // if it was started as service on NT try to stop the service + if(Service.IsNT()) + Service.Stop(); +#endif + + /* Wait until cleanup is done */ + (void) pthread_mutex_lock(&LOCK_thread_count); + while (!ready_to_exit) + { + pthread_cond_wait(&COND_thread_count,&LOCK_thread_count); + } + (void) pthread_mutex_unlock(&LOCK_thread_count); + (void) my_delete(pidfile_name,MYF(MY_WME)); // Not neaded anymore + my_thread_end(); + exit(0); + return(0); /* purecov: deadcode */ +} + + +#ifdef __WIN__ +/* ------------------------------------------------------------------------ + main and thread entry function for Win32 + (all this is needed only to run mysqld as a service on WinNT) + -------------------------------------------------------------------------- */ +int mysql_service(void *p) +{ + win_main(Service.my_argc, Service.my_argv); + return 0; +} + +int main(int argc, char **argv) +{ + // check environment variable OS + if (Service.GetOS()) // "OS" defined; Should be NT + { + if (argc == 2) + { + if (!strcmp(argv[1],"-install") || !strcmp(argv[1],"--install")) + { + char path[FN_REFLEN]; + my_path(path, argv[0], ""); // Find name in path + fn_format(path,argv[0],path,"",1+4+16); // Force use of full path + if (!Service.Install(MYSQL_SERVICENAME,MYSQL_SERVICENAME,path)) + MessageBox(NULL,"Failed to install Service",MYSQL_SERVICENAME, + MB_OK|MB_ICONSTOP); + return 0; + } + else if (!strcmp(argv[1],"-remove") || !strcmp(argv[1],"--remove")) + { + Service.Remove(MYSQL_SERVICENAME); + return 0; + } + } + else if (argc == 1) // No arguments; start as a service + { + // init service + long tmp=Service.Init(MYSQL_SERVICENAME,mysql_service); + return 0; + } + } + + // This is a WIN95 machine or a start of mysqld as a standalone program + // we have to pass the arguments, in case of NT-service this will be done + // by ServiceMain() + + Service.my_argc=argc; + Service.my_argv=argv; + mysql_service(NULL); + return 0; +} +/* ------------------------------------------------------------------------ */ +#endif + + +static int bootstrap(FILE *file) +{ + THD *thd= new THD; + int error; + thd->bootstrap=1; + thd->client_capabilities=0; + my_net_init(&thd->net,(Vio*) 0); + thd->max_packet_length=thd->net.max_packet; + thd->master_access= ~0; + thread_count++; + thd->real_id=pthread_self(); + error=handle_bootstrap(thd,file); + net_end(&thd->net); + delete thd; + return error; +} + +static bool read_init_file(char *file_name) +{ + FILE *file; + DBUG_ENTER("read_init_file"); + DBUG_PRINT("enter",("name: %s",file_name)); + if (!(file=my_fopen(file_name,O_RDONLY,MYF(MY_WME)))) + return(1); + bootstrap(file); /* Ignore errors from this */ + (void) my_fclose(file,MYF(MY_WME)); + return 0; +} + + +static void create_new_thread(THD *thd) +{ + DBUG_ENTER("create_new_thread"); + + NET *net=&thd->net; // For easy ref + net->timeout = (uint) connect_timeout; // Timeout for read + if (protocol_version > 9) + net->return_errno=1; + + /* don't allow too many connections */ + if (thread_count - delayed_insert_threads >= max_connections+1 || abort_loop) + { + DBUG_PRINT("error",("too many connections")); + close_connection(net,ER_CON_COUNT_ERROR); + delete thd; + DBUG_VOID_RETURN; + } + if (pthread_mutex_lock(&LOCK_thread_count)) + { + DBUG_PRINT("error",("Can't lock LOCK_thread_count")); + close_connection(net,ER_OUT_OF_RESOURCES); + delete thd; + DBUG_VOID_RETURN; + } + if (thread_count-delayed_insert_threads > max_used_connections) + max_used_connections=thread_count-delayed_insert_threads; + thd->thread_id=thread_id++; + for (uint i=0; i < 8 ; i++) // Generate password teststring + thd->scramble[i]= (char) (rnd(&sql_rand)*94+33); + thd->scramble[8]=0; + thd->rand=sql_rand; + thd->real_id=pthread_self(); // Keep purify happy + + /* Start a new thread to handle connection */ +#ifndef DBUG_OFF + if (test_flags & TEST_NO_THREADS) // For debugging under Linux + { + thread_cache_size=0; // Safety + thread_count++; + threads.append(thd); + thd->real_id=pthread_self(); + (void) pthread_mutex_unlock(&LOCK_thread_count); + handle_one_connection((void*) thd); + } + else +#endif + { + if (cached_thread_count > wake_thread) + { + start_cached_thread(thd); + (void) pthread_mutex_unlock(&LOCK_thread_count); + } + else + { + int error; + thread_count++; + threads.append(thd); + DBUG_PRINT("info",(("creating thread %d"), thd->thread_id)); + thd->connect_time = time(NULL); + (void) pthread_mutex_unlock(&LOCK_thread_count); + if ((error=pthread_create(&thd->real_id,&connection_attrib, + handle_one_connection, + (void*) thd))) + { + DBUG_PRINT("error", + ("Can't create thread to handle request (error %d)", + error)); + (void) pthread_mutex_lock(&LOCK_thread_count); + thread_count--; + thd->killed=1; // Safety + (void) pthread_mutex_unlock(&LOCK_thread_count); + net_printf(net,ER_CANT_CREATE_THREAD,error); + (void) pthread_mutex_lock(&LOCK_thread_count); + close_connection(net,0,0); + delete thd; + (void) pthread_mutex_unlock(&LOCK_thread_count); + DBUG_VOID_RETURN; + } + } + } + DBUG_PRINT("info",(("Thread %d created"), thd->thread_id)); + DBUG_VOID_RETURN; +} + + + /* Handle new connections and spawn new process to handle them */ + +pthread_handler_decl(handle_connections_sockets,arg __attribute__((unused))) +{ + my_socket sock,new_sock; + uint error_count=0; + uint max_used_connection= (uint) (max(ip_sock,unix_sock)+1); + fd_set readFDs,clientFDs; + THD *thd; + struct sockaddr_in cAddr; + int ip_flags=0,socket_flags=0,flags; + Vio *vio_tmp; +#ifdef __WIN__ + my_thread_init(); +#endif + DBUG_ENTER("handle_connections_sockets"); + + LINT_INIT(new_sock); + + (void) my_pthread_getprio(pthread_self()); // For debugging + + FD_ZERO(&clientFDs); + if (ip_sock != INVALID_SOCKET) + { + FD_SET(ip_sock,&clientFDs); +#ifdef HAVE_FCNTL + ip_flags = fcntl(ip_sock, F_GETFL, 0); +#endif + } +#ifdef HAVE_SYS_UN_H + FD_SET(unix_sock,&clientFDs); + socket_flags=fcntl(unix_sock, F_GETFL, 0); +#endif + + DBUG_PRINT("general",("Waiting for connections.")); + while (!abort_loop) + { + readFDs=clientFDs; +#ifdef HPUX + if (select(max_used_connection,(int*) &readFDs,0,0,0) < 0) + continue; +#else + if (select((int) max_used_connection,&readFDs,0,0,0) < 0) + { + if (errno != EINTR) + { + if (!select_errors++ && !abort_loop) /* purecov: inspected */ + sql_print_error("mysqld: Got error %d from select",errno); /* purecov: inspected */ + } + continue; + } +#endif /* HPUX */ + if (abort_loop) + break; + + /* + ** Is this a new connection request + */ + +#ifdef HAVE_SYS_UN_H + if (FD_ISSET(unix_sock,&readFDs)) + { + sock = unix_sock; + flags= socket_flags; + } + else +#endif + { + sock = ip_sock; + flags= ip_flags; + } + +#if !defined(NO_FCNTL_NONBLOCK) + if (!(test_flags & TEST_BLOCKING)) + { +#if defined(O_NONBLOCK) + fcntl(sock, F_SETFL, flags | O_NONBLOCK); +#elif defined(O_NDELAY) + fcntl(sock, F_SETFL, flags | O_NDELAY); +#endif + } +#endif /* NO_FCNTL_NONBLOCK */ + for (uint retry=0; retry < MAX_ACCEPT_RETRY; retry++) + { + size_socket length=sizeof(struct sockaddr_in); + new_sock = accept(sock, my_reinterpret_cast(struct sockaddr *) (&cAddr), + &length); + if (new_sock != INVALID_SOCKET || (errno != EINTR && errno != EAGAIN)) + break; +#if !defined(NO_FCNTL_NONBLOCK) + if (!(test_flags & TEST_BLOCKING)) + { + if (retry == MAX_ACCEPT_RETRY - 1) + fcntl(sock, F_SETFL, flags); // Try without O_NONBLOCK + } +#endif + } +#if !defined(NO_FCNTL_NONBLOCK) + if (!(test_flags & TEST_BLOCKING)) + fcntl(sock, F_SETFL, flags); +#endif + if (new_sock < 0) + { + if ((error_count++ & 255) == 0) // This can happen often + sql_perror("Error in accept"); + if (errno == ENFILE || errno == EMFILE) + sleep(1); // Give other threads some time + continue; + } + +#ifdef HAVE_LIBWRAP + { + if (sock == ip_sock) + { + struct request_info req; + signal(SIGCHLD, SIG_DFL); + request_init(&req, RQ_DAEMON, libwrapName, RQ_FILE, new_sock, NULL); + fromhost(&req); + if (!hosts_access(&req)) + { + // This may be stupid but refuse() includes an exit(0) + // which we surely don't want... + // clean_exit() - same stupid thing ... + syslog(deny_severity, "refused connect from %s", eval_client(&req)); + if (req.sink) + ((void (*)(int))req.sink)(req.fd); + + // C++ sucks (the gibberish in front just translates the supplied + // sink function pointer in the req structure from a void (*sink)(); + // to a void(*sink)(int) if you omit the cast, the C++ compiler + // will cry... + + (void) shutdown(new_sock,2); // This looks fine to me... + (void) closesocket(new_sock); + continue; + } + } + } +#endif /* HAVE_LIBWRAP */ + + { + size_socket dummyLen; + struct sockaddr dummy; + dummyLen = sizeof(struct sockaddr); + if (getsockname(new_sock,&dummy, &dummyLen) < 0) + { + sql_perror("Error on new connection socket"); + (void) shutdown(new_sock,2); + (void) closesocket(new_sock); + continue; + } + } + + /* + ** Don't allow too many connections + */ + + if (!(thd= new THD)) + { + (void) shutdown(new_sock,2); VOID(closesocket(new_sock)); + continue; + } + if (!(vio_tmp=vio_new(new_sock, + new_sock == unix_sock ? VIO_TYPE_SOCKET : + VIO_TYPE_TCPIP, + new_sock == unix_sock)) || + my_net_init(&thd->net,vio_tmp)) + { + if (vio_tmp) + vio_delete(vio_tmp); + else + { + (void) shutdown(new_sock,2); + (void) closesocket(new_sock); + } + delete thd; + continue; + } + if (sock == unix_sock) + { + if (!(thd->host=my_strdup(LOCAL_HOST,MYF(0)))) + { + close_connection(&thd->net,ER_OUT_OF_RESOURCES); + delete thd; + continue; + } + } + create_new_thread(thd); + } + +#ifdef __NT__ + pthread_mutex_lock(&LOCK_thread_count); + handler_count--; + pthread_cond_signal(&COND_handler_count); + pthread_mutex_unlock(&LOCK_thread_count); +#endif + DBUG_RETURN(0); +} + + +#ifdef __NT__ +pthread_handler_decl(handle_connections_namedpipes,arg) +{ + HANDLE hConnectedPipe; + BOOL fConnected; + THD *thd; + my_thread_init(); + DBUG_ENTER("handle_connections_namedpipes"); + (void) my_pthread_getprio(pthread_self()); // For debugging + + DBUG_PRINT("general",("Waiting for named pipe connections.")); + while (!abort_loop) + { + /* wait for named pipe connection */ + fConnected = ConnectNamedPipe( hPipe, NULL ); + if (abort_loop) + break; + if ( !fConnected ) + fConnected = GetLastError() == ERROR_PIPE_CONNECTED; + if ( !fConnected ) + { + CloseHandle( hPipe ); + if ((hPipe = CreateNamedPipe(szPipeName, + PIPE_ACCESS_DUPLEX, + PIPE_TYPE_BYTE | + PIPE_READMODE_BYTE | + PIPE_WAIT, + PIPE_UNLIMITED_INSTANCES, + (int) net_buffer_length, + (int) net_buffer_length, + NMPWAIT_USE_DEFAULT_WAIT, + &saPipeSecurity )) == + INVALID_HANDLE_VALUE ) + { + sql_perror("Can't create new named pipe!"); + break; // Abort + } + } + hConnectedPipe = hPipe; + /* create new pipe for new connection */ + if ((hPipe = CreateNamedPipe(szPipeName, + PIPE_ACCESS_DUPLEX, + PIPE_TYPE_BYTE | + PIPE_READMODE_BYTE | + PIPE_WAIT, + PIPE_UNLIMITED_INSTANCES, + (int) net_buffer_length, + (int) net_buffer_length, + NMPWAIT_USE_DEFAULT_WAIT, + &saPipeSecurity)) == + INVALID_HANDLE_VALUE) + { + sql_perror("Can't create new named pipe!"); + hPipe=hConnectedPipe; + continue; // We have to try again + } + + if ( !(thd = new THD)) + { + DisconnectNamedPipe( hConnectedPipe ); + CloseHandle( hConnectedPipe ); + continue; + } + if (!(thd->net.vio = vio_new_win32pipe(hConnectedPipe)) || + my_net_init(&thd->net, thd->net.vio)) + { + close_connection(&thd->net,ER_OUT_OF_RESOURCES); + delete thd; + continue; + } + /* host name is unknown */ + thd->host = my_strdup("localhost",MYF(0)); /* Host is unknown */ + create_new_thread(thd); + } + + pthread_mutex_lock(&LOCK_thread_count); + handler_count--; + pthread_cond_signal(&COND_handler_count); + pthread_mutex_unlock(&LOCK_thread_count); + DBUG_RETURN(0); +} +#endif /* __NT__ */ + +/**************************************************************************** +** Create thread that automaticly flush all tables after a given time +****************************************************************************/ + +pthread_handler_decl(handle_flush,arg __attribute__((unused))) +{ + my_thread_init(); + DBUG_ENTER("handle_flush"); + + pthread_detach_this_thread(); + flush_thread=pthread_self(); + flush_thread_in_use=1; + + pthread_mutex_lock(&LOCK_flush); + for (;;) + { + struct timespec abstime; +#ifdef HAVE_TIMESPEC_TS_SEC + abstime.ts_sec=time(NULL)+flush_time; // Bsd 2.1 + abstime.ts_nsec=0; +#else + abstime.tv_sec=time(NULL)+flush_time; // Linux or Solairs + abstime.tv_nsec=0; +#endif + (void) pthread_cond_timedwait(&COND_flush,&LOCK_flush, &abstime); + if (abort_loop) + break; + flush_tables(); + } + flush_thread_in_use=0; + pthread_mutex_unlock(&LOCK_flush); + my_thread_end(); + DBUG_RETURN(0); +} + + +/****************************************************************************** +** handle start options +******************************************************************************/ + +enum options {OPT_ISAM_LOG=256,OPT_SKIP_NEW,OPT_SKIP_GRANT, + OPT_SKIP_LOCK,OPT_ENABLE_LOCK, + OPT_USE_LOCKING,OPT_SOCKET,OPT_UPDATE_LOG, OPT_BIN_LOG, + OPT_SKIP_RESOLVE,OPT_SKIP_NETWORKING, OPT_BIN_LOG_INDEX, + OPT_BIND_ADDRESS,OPT_PID_FILE,OPT_SKIP_PRIOR,OPT_BIG_TABLES, + OPT_STANDALONE,OPT_ONE_THREAD,OPT_CONSOLE, + OPT_LOW_PRIORITY_UPDATES,OPT_SKIP_HOST_CACHE,OPT_LONG_FORMAT, + OPT_FLUSH, OPT_SAFE, OPT_BOOTSTRAP, OPT_SKIP_SHOW_DB, + OPT_TABLE_TYPE, OPT_INIT_FILE, OPT_DELAY_KEY_WRITE, + OPT_SLOW_QUERY_LOG, OPT_SKIP_DELAY_KEY_WRITE, + OPT_CHARSETS_DIR, + OPT_BDB_HOME, OPT_BDB_LOG, OPT_BDB_TMP, OPT_BDB_NOSYNC, + OPT_BDB_LOCK, OPT_BDB_SKIP, OPT_BDB_RECOVER, + OPT_MASTER_HOST, OPT_MASTER_USER, OPT_MASTER_PASSWORD, + OPT_MASTER_PORT, OPT_MASTER_INFO_FILE, OPT_MASTER_CONNECT_RETRY, + OPT_SQL_BIN_UPDATE_SAME, OPT_REPLICATE_DO_DB, + OPT_REPLICATE_IGNORE_DB, OPT_LOG_SLAVE_UPDATES, + OPT_BINLOG_DO_DB, OPT_BINLOG_IGNORE_DB}; + +static struct option long_options[] = +{ + {"ansi", no_argument,0, 'a'}, + {"basedir", required_argument, 0, 'b'}, +#ifdef HAVE_BERKELEY_DB + {"bdb-home", required_argument, 0, OPT_BDB_HOME}, + {"bdb-lock-detect", required_argument, 0, OPT_BDB_LOCK}, + {"bdb-logdir", required_argument, 0, OPT_BDB_LOG}, + {"bdb-recover", no_argument, 0, OPT_BDB_RECOVER}, + {"bdb-no-sync", no_argument, 0, OPT_BDB_NOSYNC}, + {"bdb-tmpdir", required_argument, 0, OPT_BDB_TMP}, +#endif + {"big-tables", no_argument,0,(int) OPT_BIG_TABLES}, + {"binlog-do-db", required_argument, 0, OPT_BINLOG_DO_DB}, + {"binlog-ignore-db", required_argument, 0, OPT_BINLOG_IGNORE_DB}, + {"bind-address", required_argument, 0, OPT_BIND_ADDRESS}, + {"bootstrap", no_argument,0,(int) OPT_BOOTSTRAP}, +#ifdef __WIN__ + {"console", no_argument, 0, OPT_CONSOLE}, +#endif + {"chroot", required_argument,0, 'r'}, + {"character-sets-dir",required_argument,0, OPT_CHARSETS_DIR}, + {"datadir", required_argument, 0, 'h'}, +#ifndef DBUG_OFF + {"debug", optional_argument, 0, '#'}, +#endif + {"default-character-set",required_argument,0,'C'}, + {"default-table-type",required_argument,0,OPT_TABLE_TYPE}, + {"delay-key-write-for-all-tables", + no_argument, 0, (int) OPT_DELAY_KEY_WRITE}, + {"enable-locking", no_argument, 0, (int) OPT_ENABLE_LOCK}, + {"exit-info", optional_argument, 0, 'T'}, + {"flush", no_argument, 0, OPT_FLUSH}, + {"help", no_argument, 0, '?'}, + {"init-file", required_argument, 0, (int) OPT_INIT_FILE}, + {"log", optional_argument, 0, 'l'}, + {"language", required_argument, 0, 'L'}, + {"log-bin", optional_argument, 0, (int) OPT_BIN_LOG}, + {"log-bin-index", optional_argument, 0, (int) OPT_BIN_LOG_INDEX}, + {"log-isam", optional_argument, 0, (int) OPT_ISAM_LOG}, + {"log-update", optional_argument, 0, (int) OPT_UPDATE_LOG}, + {"log-slow-queries", optional_argument, 0, (int) OPT_SLOW_QUERY_LOG}, + {"log-long-format", no_argument, 0, (int) OPT_LONG_FORMAT}, + {"log-slave-updates", no_argument,0, (int) OPT_LOG_SLAVE_UPDATES}, + {"low-priority-updates", no_argument, 0, (int) OPT_LOW_PRIORITY_UPDATES}, + {"master-host", required_argument, 0, (int) OPT_MASTER_HOST}, + {"master-user", required_argument, 0, (int) OPT_MASTER_USER}, + {"master-password", required_argument, 0, (int) OPT_MASTER_PASSWORD}, + {"master-port", required_argument, 0, (int) OPT_MASTER_PORT}, + {"master-connect-retry", required_argument, 0, (int) OPT_MASTER_CONNECT_RETRY}, + {"master-info-file", required_argument, 0, (int) OPT_MASTER_INFO_FILE}, + {"new", no_argument, 0, 'n'}, + {"old-protocol", no_argument, 0, 'o'}, + {"one-thread", no_argument, 0, OPT_ONE_THREAD}, + {"pid-file", required_argument, 0, (int) OPT_PID_FILE}, + {"port", required_argument, 0, 'P'}, + {"replicate-do-db", required_argument, 0, OPT_REPLICATE_DO_DB}, + {"replicate-ignore-db", required_argument, 0, OPT_REPLICATE_IGNORE_DB}, + {"safe-mode", no_argument, 0, (int) OPT_SAFE}, + {"socket", required_argument, 0, (int) OPT_SOCKET}, + {"set-variable", required_argument, 0, 'O'}, +#ifdef HAVE_BERKELEY_DB + {"skip-bdb", no_argument, 0, OPT_BDB_SKIP}, +#endif + {"skip-delay-key-write", no_argument, 0, (int) OPT_SKIP_DELAY_KEY_WRITE}, + {"skip-grant-tables", no_argument,0, (int) OPT_SKIP_GRANT}, + {"skip-locking", no_argument,0, (int) OPT_SKIP_LOCK}, + {"skip-host-cache", no_argument,0, (int) OPT_SKIP_HOST_CACHE}, + {"skip-name-resolve", no_argument,0, (int) OPT_SKIP_RESOLVE}, + {"skip-new", no_argument,0, (int) OPT_SKIP_NEW}, + {"skip-show-database",no_argument,0, (int) OPT_SKIP_SHOW_DB}, + {"skip-networking", no_argument,0, (int) OPT_SKIP_NETWORKING}, + {"skip-thread-priority", no_argument,0,(int) OPT_SKIP_PRIOR}, + {"sql-bin-update-same", no_argument, 0, (int)OPT_SQL_BIN_UPDATE_SAME}, +#include "sslopt-longopts.h" +#ifdef __WIN__ + {"standalone", no_argument,0, (int) OPT_STANDALONE}, +#endif + {"tmpdir", required_argument, 0, 't'}, + {"use-locking", no_argument, 0,(int) OPT_USE_LOCKING}, +#ifdef USE_SYMDIR + {"use-symbolic-links",no_argument, 0, 's'}, +#endif + {"user", required_argument, 0, 'u'}, + {"version", no_argument, 0, 'V'}, + {0, 0, 0, 0} +}; + +CHANGEABLE_VAR changeable_vars[] = { + { "back_log", (long*) &back_log,50,1,65535,0,1}, +#ifdef HAVE_BERKELEY_DB + { "bdb_cache_size", (long*) &berkeley_cache_size, KEY_CACHE_SIZE, + 20*1024, (long) ~0, 0, IO_SIZE}, +#endif + { "connect_timeout", (long*) &connect_timeout,CONNECT_TIMEOUT,1,65535,0,1}, + { "delayed_insert_timeout",(long*) &delayed_insert_timeout, + DELAYED_WAIT_TIMEOUT,1,~0L,0,1}, + { "delayed_insert_limit",(long*) &delayed_insert_limit, + DELAYED_LIMIT,1,~0L,0,1}, + { "delayed_queue_size", (long*) &delayed_queue_size,DELAYED_QUEUE_SIZE,1, + ~0L,0,1}, + { "flush_time", (long*) &flush_time,FLUSH_TIME,0,~0L,0,1}, + { "interactive_timeout", (long*) &net_interactive_timeout, + NET_WAIT_TIMEOUT,1,31*24*60*60,0,1}, + { "join_buffer_size", (long*) &join_buff_size,128*1024L, + IO_SIZE*2+MALLOC_OVERHEAD,~0L,MALLOC_OVERHEAD,IO_SIZE }, + { "key_buffer_size", (long*) &keybuff_size,KEY_CACHE_SIZE,MALLOC_OVERHEAD, + (long) ~0, MALLOC_OVERHEAD, IO_SIZE }, + { "long_query_time", (long*) &long_query_time,10,1,~0L,0,1}, + { "lower_case_table_names", (long*) &lower_case_table_names,IF_WIN(1,0), + 0,1,0,1}, + { "max_allowed_packet",(long*) &max_allowed_packet,1024*1024L,80, + 17*1024*1024L, MALLOC_OVERHEAD,1024}, + { "max_connections", (long*) &max_connections,100,1,16384,0,1}, + { "max_connect_errors", (long*) &max_connect_errors,MAX_CONNECT_ERRORS,1, + ~0L,0,1}, + { "max_delayed_threads", (long*) &max_insert_delayed_threads,20,1, + 16384,0,1}, + { "max_heap_table_size",(long*) &max_heap_table_size,16*1024*1024L,16384,~0L, + MALLOC_OVERHEAD,1024}, + { "max_join_size",(long*) &max_join_size,~0L,1,~0L,0,1}, + { "max_sort_length",(long*) &max_item_sort_length,1024,4,8192*1024L,0,1}, + { "max_tmp_tables",(long*) &max_tmp_tables,32,1,~0L,0,1}, + { "max_write_lock_count",(long*) &max_write_lock_count,~0L,1,~0L,0,1}, + { "myisam_sort_buffer_size", (long*) &myisam_sort_buffer_size,8192*1024,4, + ~0L,0,1}, + { "net_buffer_length",(long*) &net_buffer_length,16384,1024,1024*1024L, + MALLOC_OVERHEAD,1024}, + { "net_retry_count",(long*) &mysqld_net_retry_count,MYSQLD_NET_RETRY_COUNT, + 1,~0L, 0,1}, + { "net_read_timeout", (long*) &net_read_timeout, NET_READ_TIMEOUT,1,65535,0,1}, + { "net_write_timeout", (long*) &net_write_timeout,NET_WRITE_TIMEOUT,1,65535,0,1}, + { "query_buffer_size", (long*) &query_buff_size,0,MALLOC_OVERHEAD, + (long) ~0, MALLOC_OVERHEAD, IO_SIZE }, + { "record_buffer", (long*) &my_default_record_cache_size,128*1024L, + IO_SIZE*2+MALLOC_OVERHEAD,~0L,MALLOC_OVERHEAD,IO_SIZE }, + { "slow_launch_time", (long*) &slow_launch_time, 2L, + 0L,~0L,0,1 }, + { "sort_buffer", (long*) &sortbuff_size,MAX_SORT_MEMORY, + MIN_SORT_MEMORY+MALLOC_OVERHEAD*2,~0L,MALLOC_OVERHEAD,1 }, + { "table_cache", (long*) &table_cache_size,64,1,16384,0,1}, + { "thread_concurrency", (long*) &concurrency,DEFAULT_CONCURRENCY,1,512,0,1}, + { "thread_cache_size", (long*) &thread_cache_size, 0,1,16384,0,1}, + { "tmp_table_size", (long*) &tmp_table_size,1024*1024L,1024,~0L, + MALLOC_OVERHEAD,1}, + { "thread_stack", (long*) &thread_stack,1024*64,1024*32,~0L,0,1024}, + { "wait_timeout", (long*) &net_wait_timeout,NET_WAIT_TIMEOUT,1,~0L,0,1}, + { NullS,(long*) 0,0,0,0,0,0,} }; + + +struct show_var_st init_vars[]= { + {"ansi_mode",(char*) &opt_ansi_mode ,SHOW_BOOL}, + {"back_log", (char*) &back_log, SHOW_LONG}, + {"basedir", mysql_home,SHOW_CHAR}, +#ifdef HAVE_BERKELEY_DB + {"bdb_cache_size", (char*) &berkeley_cache_size, SHOW_LONG}, + {"bdb_home", (char*) &berkeley_home, SHOW_CHAR_PTR}, + {"bdb_logdir", (char*) &berkeley_logdir, SHOW_CHAR_PTR}, + {"bdb_tmpdir", (char*) &berkeley_tmpdir, SHOW_CHAR_PTR}, +#endif + {"character_set", default_charset, SHOW_CHAR}, + {"character_sets", (char*) &charsets_list, SHOW_CHAR_PTR}, + {"connect_timeout", (char*) &connect_timeout, SHOW_LONG}, + {"concurrent_insert",(char*) &myisam_concurrent_insert, SHOW_MY_BOOL}, + {"datadir", mysql_real_data_home,SHOW_CHAR}, + {"delay_key_write",(char*) &myisam_delay_key_write, SHOW_MY_BOOL}, + {"delayed_insert_limit",(char*) &delayed_insert_limit,SHOW_LONG}, + {"delayed_insert_timeout",(char*) &delayed_insert_timeout,SHOW_LONG}, + {"delayed_queue_size", (char*) &delayed_queue_size,SHOW_LONG}, + {"join_buffer_size", (char*) &join_buff_size, SHOW_LONG}, + {"flush", (char*) &myisam_flush, SHOW_MY_BOOL}, + {"flush_time",(char*) &flush_time, SHOW_LONG}, + {"init_file", (char*) &opt_init_file, SHOW_CHAR_PTR}, + {"interactive_timeout", (char*) &net_interactive_timeout, SHOW_LONG}, + {"key_buffer_size", (char*) &keybuff_size, SHOW_LONG}, + {"language", language, SHOW_CHAR}, + {"log", (char*) &opt_log, SHOW_BOOL}, + {"log_update",(char*) &opt_update_log,SHOW_BOOL}, + {"log_bin",(char*) &opt_bin_log,SHOW_BOOL}, + {"log_slave_updates",(char*) &opt_log_slave_updates,SHOW_BOOL}, + {"long_query_time", (char*) &long_query_time, SHOW_LONG}, + {"low_priority_updates", (char*) &low_priority_updates, SHOW_BOOL}, + {"lower_case_table_names", (char*) &lower_case_table_names, SHOW_LONG}, + {"max_allowed_packet", (char*) &max_allowed_packet, SHOW_LONG}, + {"max_connections", (char*) &max_connections, SHOW_LONG}, + {"max_connect_errors", (char*) &max_connect_errors, SHOW_LONG}, + {"max_delayed_threads", (char*) &max_insert_delayed_threads, SHOW_LONG}, + {"max_heap_table_size", (char*) &max_heap_table_size,SHOW_LONG}, + {"max_join_size", (char*) &max_join_size, SHOW_LONG}, + {"max_sort_length", (char*) &max_item_sort_length,SHOW_LONG}, + {"max_tmp_tables", (char*) &max_tmp_tables,SHOW_LONG}, + {"max_write_lock_count",(char*) &max_write_lock_count,SHOW_LONG}, + {"myisam_sort_buffer_size", (char*) &myisam_sort_buffer_size,SHOW_LONG}, + {"net_buffer_length", (char*) &net_buffer_length, SHOW_LONG}, + {"net_retry_count", (char*) &mysqld_net_retry_count, SHOW_LONG}, + {"pid_file", (char*) pidfile_name, SHOW_CHAR}, + {"port", (char*) &mysql_port, SHOW_INT}, + {"protocol_version", (char*) &protocol_version, SHOW_INT}, + {"record_buffer", (char*) &my_default_record_cache_size,SHOW_LONG}, + {"skip_locking", (char*) &my_disable_locking, SHOW_MY_BOOL}, + {"skip_networking", (char*) &opt_disable_networking, SHOW_BOOL}, + {"skip_show_database", (char*) &opt_skip_show_db, SHOW_BOOL}, + {"slow_launch_time", (char*) &slow_launch_time, SHOW_LONG}, + {"socket", (char*) &mysql_unix_port, SHOW_CHAR_PTR}, + {"sort_buffer", (char*) &sortbuff_size,SHOW_LONG}, + {"table_cache", (char*) &table_cache_size,SHOW_LONG}, + {"table_type", (char*) (&default_table_type_name), SHOW_CHAR_PTR}, +#ifdef HAVE_THR_SETCONCURRENCY + {"thread_concurrency", (char*) &concurrency, SHOW_LONG}, +#endif + {"thread_stack", (char*) &thread_stack,SHOW_LONG}, + {"thread_cache_size", (char*) &thread_cache_size,SHOW_LONG}, +#ifdef HAVE_TZNAME + {"timezone", time_zone, SHOW_CHAR}, +#endif + {"tmp_table_size", (char*) &tmp_table_size,SHOW_LONG}, + {"tmpdir", (char*) &mysql_tmpdir,SHOW_CHAR_PTR}, + {"version", server_version, SHOW_CHAR}, + {"wait_timeout",(char*) &net_wait_timeout,SHOW_LONG}, + {NullS,NullS,SHOW_LONG} +}; + +struct show_var_st status_vars[]= { + {"Aborted_clients", (char*) &aborted_threads, SHOW_LONG}, + {"Aborted_connects", (char*) &aborted_connects, SHOW_LONG}, + {"Bytes_received", (char*) &bytes_received, SHOW_LONG}, + {"Bytes_sent", (char*) &bytes_sent, SHOW_LONG}, + {"Connections", (char*) &thread_id, SHOW_LONG_CONST}, + {"Created_tmp_tables",(char*) &created_tmp_tables, SHOW_LONG}, + {"Delayed_insert_threads", (char*) &delayed_insert_threads, SHOW_LONG}, + {"Delayed_writes", (char*) &delayed_insert_writes, SHOW_LONG}, + {"Delayed_errors", (char*) &delayed_insert_errors, SHOW_LONG}, + {"Flush_commands", (char*) &refresh_version, SHOW_LONG_CONST}, + {"Handler_delete", (char*) &ha_delete_count, SHOW_LONG}, + {"Handler_read_first",(char*) &ha_read_first_count, SHOW_LONG}, + {"Handler_read_key", (char*) &ha_read_key_count, SHOW_LONG}, + {"Handler_read_next", (char*) &ha_read_next_count, SHOW_LONG}, + {"Handler_read_prev", (char*) &ha_read_prev_count, SHOW_LONG}, + {"Handler_read_rnd", (char*) &ha_read_rnd_count, SHOW_LONG}, + {"Handler_read_rnd_next", (char*) &ha_read_rnd_next_count, SHOW_LONG}, + {"Handler_update", (char*) &ha_update_count, SHOW_LONG}, + {"Handler_write", (char*) &ha_write_count, SHOW_LONG}, + {"Key_blocks_used", (char*) &_my_blocks_used, SHOW_LONG_CONST}, + {"Key_read_requests", (char*) &_my_cache_r_requests, SHOW_LONG}, + {"Key_reads", (char*) &_my_cache_read, SHOW_LONG}, + {"Key_write_requests",(char*) &_my_cache_w_requests, SHOW_LONG}, + {"Key_writes", (char*) &_my_cache_write, SHOW_LONG}, + {"Max_used_connections", (char*) &max_used_connections, SHOW_LONG}, + {"Not_flushed_key_blocks", (char*) &_my_blocks_changed, SHOW_LONG_CONST}, + {"Not_flushed_delayed_rows", (char*) &delayed_rows_in_use, SHOW_LONG_CONST}, + {"Open_tables", (char*) 0, SHOW_OPENTABLES}, + {"Open_files", (char*) &my_file_opened, SHOW_INT_CONST}, + {"Open_streams", (char*) &my_stream_opened, SHOW_INT_CONST}, + {"Opened_tables", (char*) &opened_tables, SHOW_LONG}, + {"Questions", (char*) 0, SHOW_QUESTION}, + {"Slow_launch_threads", (char*) &slow_launch_threads, SHOW_LONG}, + {"Slow_queries", (char*) &long_query_count, SHOW_LONG}, + {"Slave_running", (char*) &slave_running, SHOW_BOOL}, + {"Threads_cached", (char*) &cached_thread_count, SHOW_LONG_CONST}, + {"Threads_connected", (char*) &thread_count, SHOW_INT_CONST}, + {"Threads_running", (char*) &thread_running, SHOW_INT_CONST}, + {"Uptime", (char*) 0, SHOW_STARTTIME}, + {NullS,NullS,SHOW_LONG} +}; + +static void print_version(void) +{ + printf("%s Ver %s for %s on %s\n",my_progname, + server_version,SYSTEM_TYPE,MACHINE_TYPE); +} + +static void usage(void) +{ + print_version(); + puts("Copyright (C) 2000 MySQL AB & MySQL Finland AB, by Monty and others"); + puts("This software comes with ABSOLUTELY NO WARRANTY. This is free software,\nand you are welcome to modify and redistribute it under the GPL license\n"); + puts("Starts the MySQL server\n"); + + printf("Usage: %s [OPTIONS]\n", my_progname); + puts("\n\ + --ansi Use ANSI SQL syntax instead of MySQL syntax\n\ + -b, --basedir=path Path to installation directory. All paths are\n\ + usually resolved relative to this\n\ + --big-tables Allow big result sets by saving all temporary sets\n\ + on file (Solves most 'table full' errors)\n\ + --bind-address=IP Ip address to bind to\n\ + --bootstrap Used by mysql installation scripts\n\ + --character-sets-dir=...\n\ + Directory where character sets are\n\ + --chroot=path Chroot mysqld daemon during startup\n\ + -h, --datadir=path Path to the database root"); +#ifndef DBUG_OFF + printf("\ + -#, --debug[=...] Debug log. Default is '%s'\n",default_dbug_option); +#endif + puts("\ + --default-character-set=charset\n\ + Set the default character set\n\ + --default-table-type=type\n\ + Set the default table type for tables\n\ + --delay-key-write-for-all-tables\n\ + Don't flush key buffers between writes for any MyISAM\n\ + table\n\ + --enable-locking Enable system locking\n\ + -T, --exit-info Print some debug info at exit\n\ + --flush Flush tables to disk between SQL commands\n\ + -?, --help Display this help and exit\n\ + --init-file=file Read SQL commands from this file at startup\n\ + -L, --language=... Client error messages in given language. May be\n\ + given as a full path\n\ + -l, --log[=file] Log connections and queries to file\n\ + --log-bin[=file] Log queries in new binary format (for replication)\n\ + --log-bin-index=file File that holds the names for last binary log files\n\ + --log-update[=file] Log updates to file.# where # is a unique number\n\ + if not given.\n\ + --log-isam[=file] Log all isam changes to file\n\ + --log-long-format Log some extra information to update log\n\ + --low-priority-updates INSERT/DELETE/UPDATE has lower priority than selects\n\ + --pid-file=path Pid file used by safe_mysqld\n\ + -P, --port=... Port number to use for connection\n\ + -n, --new Use very new possible 'unsafe' functions\n\ + -o, --old-protocol Use the old (3.20) protocol\n\ + --one-thread Only use one thread (for debugging under Linux)\n\ + -O, --set-variable var=option\n\ + Give a variable an value. --help lists variables\n\ + -Sg, --skip-grant-tables\n\ + Start without grant tables. This gives all users\n\ + FULL ACCESS to all tables!\n\ + --safe-mode Skip some optimize stages (for testing)\n\ + --skip-delay-key-write\n\ + Ignore the delay_key_write option for all tables\n\ + --skip-locking Don't use system locking. To use isamchk one has\n\ + to shut down the server.\n\ + --skip-name-resolve Don't resolve hostnames.\n\ + All hostnames are IP's or 'localhost'\n\ + --skip-networking Don't allow connection with TCP/IP.\n\ + --skip-new Don't use new, possible wrong routines.\n\ + --skip-host-cache Don't cache host names\n"); + /* We have to break the string here because of VC++ limits */ + puts("\ + --skip-show-database Don't allow 'SHOW DATABASE' commands\n\ + --skip-thread-priority\n\ + Don't give threads different priorities.\n\ + --socket=... Socket file to use for connection\n\ + -t, --tmpdir=path Path for temporary files\n\ + -u, --user=user_name Run mysqld daemon as user\n\ + -V, --version output version information and exit"); +#ifdef __WIN__ + puts("NT and Win32 specific options:\n\ + --console Don't remove the console window\n\ + --install Install mysqld as a service (NT)\n\ + --remove Remove mysqld from the service list (NT)\n\ + --standalone Dummy option to start as a standalone program (NT)\n\ +"); +#endif +#ifdef HAVE_BERKELEY_DB + puts("\ + --bdb-home= directory Berkeley home direcory\n\ + --bdb-lock-detect=# Berkeley lock detect\n\ + (DEFAULT, OLDEST, RANDOM or YOUNGEST, # sec)\n\ + --bdb-logdir=directory Berkeley DB log file directory\n\ + --bdb-nosync Don't synchronously flush logs\n\ + --bdb-recover Start Berkeley DB in recover mode\n\ + --bdb-tmpdir=directory Berkeley DB tempfile name\n\ + --skip-bdb Don't use berkeley db (will save memory)\n\ +"); +#endif + print_defaults("my",load_default_groups); + puts(""); + +#include "sslopt-usage.h" + + fix_paths(); + set_ports(); + printf("\ +To see what values a running MySQL server is using, type\n\ +'mysqladmin variables' instead of 'mysqld --help'.\n\ +The default values (after parsing the command line arguments) are:\n\n"); + + printf("basedir: %s\n",mysql_home); + printf("datadir: %s\n",mysql_real_data_home); + printf("tmpdir: %s\n",mysql_tmpdir); + printf("language: %s\n",language); + printf("pid file: %s\n",pidfile_name); + if (opt_logname) + printf("logfile: %s\n",opt_logname); + if (opt_update_logname) + printf("update log: %s\n",opt_update_logname); + if (opt_bin_logname) + { + printf("binary log: %s\n",opt_bin_logname); + printf("binary log index: %s\n",opt_binlog_index_name); + } + if (opt_slow_logname) + printf("update log: %s\n",opt_slow_logname); + printf("TCP port: %d\n",mysql_port); +#if defined(HAVE_SYS_UN_H) && !defined(HAVE_mit_thread) + printf("Unix socket: %s\n",mysql_unix_port); +#endif + if (my_disable_locking) + puts("\nsystem locking is not in use"); + if (opt_noacl) + puts("\nGrant tables are not used. All users have full access rights"); + printf("\nPossible variables for option --set-variable (-O) are:\n"); + for (uint i=0 ; changeable_vars[i].name ; i++) + printf("%-20s current value: %lu\n", + changeable_vars[i].name, + (ulong) *changeable_vars[i].varptr); +} + + +static void set_options(void) +{ + set_all_changeable_vars( changeable_vars ); +#if !defined( my_pthread_setprio ) && !defined( HAVE_PTHREAD_SETSCHEDPARAM ) + opt_specialflag |= SPECIAL_NO_PRIOR; +#endif + + (void) strmov( default_charset, MYSQL_CHARSET); + (void) strmov( language, LANGUAGE); + (void) strmov( mysql_real_data_home, get_relative_path(DATADIR)); +#ifdef __WIN__ + /* Allow Win32 users to move MySQL anywhere */ + { + char prg_dev[LIBLEN]; + my_path(prg_dev,my_progname,"mysql/bin"); + strcat(prg_dev,"/../"); // Remove 'bin' to get base dir + cleanup_dirname(mysql_home,prg_dev); + } +#else + const char *tmpenv; + if ( !(tmpenv = getenv("MY_BASEDIR_VERSION"))) + tmpenv = DEFAULT_MYSQL_HOME; + (void) strmov( mysql_home, tmpenv ); +#endif + +#if defined( HAVE_mit_thread ) || defined( __WIN__ ) || defined( HAVE_LINUXTHREADS ) + my_disable_locking = 1; +#endif + my_bind_addr = htonl( INADDR_ANY ); +} + + /* Initiates DEBUG - but no debugging here ! */ + +static void get_options(int argc,char **argv) +{ + int c,option_index=0; + + myisam_delay_key_write=1; // Allow use of this + while ((c=getopt_long(argc,argv,"ab:C:h:#::T::?l::L:O:P:sS::t:u:noVvI?", + long_options, &option_index)) != EOF) + { + switch(c) { +#ifndef DBUG_OFF + case '#': + DBUG_PUSH(optarg ? optarg : default_dbug_option); + opt_endinfo=1; /* unireg: memory allocation */ + break; +#endif + case 'a': + opt_ansi_mode=1; + thd_startup_options|=OPTION_ANSI_MODE; + break; + case 'b': + strmov(mysql_home,optarg); + break; + case 'l': + opt_log=1; + opt_logname=optarg; // Use hostname.log if null + break; + case 'h': + strmov(mysql_real_data_home,optarg); + break; + case 'L': + strmov(language,optarg); + break; + case 'n': + opt_specialflag|= SPECIAL_NEW_FUNC; + break; + case 'o': + protocol_version=PROTOCOL_VERSION-1; + break; + case 'O': + if (set_changeable_var(optarg, changeable_vars)) + { + usage(); + exit(1); + } + break; + case 'P': + mysql_port= (unsigned int) atoi(optarg); + break; + case OPT_SOCKET: + mysql_unix_port= optarg; + break; + case 'r': + mysqld_chroot=optarg; + break; +#ifdef USE_SYMDIR + case 's': + my_use_symdir=1; /* Use internal symbolic links */ + break; +#endif + case 't': + mysql_tmpdir=optarg; + break; + case 'u': + mysqld_user=optarg; + break; + case 'v': + case 'V': + print_version(); + exit(0); + case 'I': + case '?': + usage(); + exit(0); + case 'T': + test_flags= optarg ? (uint) atoi(optarg) : (uint) ~0; + opt_endinfo=1; + break; + case 'S': + if (!optarg) + opt_specialflag|= SPECIAL_NO_NEW_FUNC | SPECIAL_SAFE_MODE; + else if (!strcmp(optarg,"l")) + my_disable_locking=1; + else if (!strcmp(optarg,"g")) + opt_noacl=1; + else + { + usage(); + exit(1); + } + break; + case (int) OPT_BIG_TABLES: + thd_startup_options|=OPTION_BIG_TABLES; + break; + case (int) OPT_ISAM_LOG: + if (optarg) + nisam_log_filename=optarg; + (void) nisam_log(1); + break; + case (int) OPT_UPDATE_LOG: + opt_update_log=1; + opt_update_logname=optarg; // Use hostname.# if null + break; + case (int) OPT_BIN_LOG_INDEX: + opt_binlog_index_name = optarg; + break; + case (int) OPT_BIN_LOG: + opt_bin_log=1; + opt_bin_logname=optarg; + break; + case (int) OPT_LOG_SLAVE_UPDATES: + opt_log_slave_updates = 1; + break; + + case (int)OPT_REPLICATE_IGNORE_DB: + { + i_string *db = new i_string(optarg); + replicate_ignore_db.push_back(db); + break; + } + case (int)OPT_REPLICATE_DO_DB: + { + i_string *db = new i_string(optarg); + replicate_do_db.push_back(db); + break; + } + + case (int)OPT_BINLOG_IGNORE_DB: + { + i_string *db = new i_string(optarg); + binlog_ignore_db.push_back(db); + break; + } + case (int)OPT_BINLOG_DO_DB: + { + i_string *db = new i_string(optarg); + binlog_do_db.push_back(db); + break; + } + + + case (int) OPT_SQL_BIN_UPDATE_SAME: + opt_sql_bin_update = 1; + break; + case (int) OPT_SLOW_QUERY_LOG: + opt_slow_log=1; + opt_slow_logname=optarg; + break; + case (int) OPT_SKIP_NEW: + opt_specialflag|= SPECIAL_NO_NEW_FUNC; + default_table_type=DB_TYPE_ISAM; + myisam_delay_key_write=0; + myisam_concurrent_insert=0; + break; + case (int) OPT_SAFE: + opt_specialflag|= SPECIAL_SAFE_MODE; + myisam_delay_key_write=0; + myisam_concurrent_insert=0; + break; + case (int) OPT_SKIP_PRIOR: + opt_specialflag|= SPECIAL_NO_PRIOR; + break; + case (int) OPT_SKIP_GRANT: + opt_noacl=1; + break; + case (int) OPT_SKIP_LOCK: + my_disable_locking=1; + break; + case (int) OPT_SKIP_HOST_CACHE: + opt_specialflag|= SPECIAL_NO_HOST_CACHE; + break; + case (int) OPT_ENABLE_LOCK: + my_disable_locking=0; + break; + case (int) OPT_USE_LOCKING: + my_disable_locking=0; + break; + case (int) OPT_SKIP_RESOLVE: + opt_specialflag|=SPECIAL_NO_RESOLVE; + break; + case (int) OPT_LONG_FORMAT: + opt_specialflag|=SPECIAL_LONG_LOG_FORMAT; + break; + case (int) OPT_SKIP_NETWORKING: + opt_disable_networking=1; + mysql_port=0; + break; + case (int) OPT_SKIP_SHOW_DB: + opt_skip_show_db=1; + opt_specialflag|=SPECIAL_SKIP_SHOW_DB; + mysql_port=0; + break; + case (int) OPT_ONE_THREAD: + test_flags |= TEST_NO_THREADS; + break; + case (int) OPT_BIND_ADDRESS: + if (optarg && isdigit(optarg[0])) + { + my_bind_addr = (ulong) inet_addr(optarg); + } + else + { + struct hostent *ent; + if (!optarg || !optarg[0]) + ent=gethostbyname(optarg); + else + { + char myhostname[255]; + if (gethostname(myhostname,sizeof(myhostname)) < 0) + { + sql_perror("Can't start server: cannot get my own hostname!"); + exit(1); + } + ent=gethostbyname(myhostname); + } + if (!ent) + { + sql_perror("Can't start server: cannot resolve hostname!"); + exit(1); + } + my_bind_addr = (ulong) ((in_addr*)ent->h_addr_list[0])->s_addr; + } + break; + case (int) OPT_PID_FILE: + strmov(pidfile_name,optarg); + break; + case (int) OPT_INIT_FILE: + opt_init_file=optarg; + break; +#ifdef __WIN__ + case (int) OPT_STANDALONE: /* Dummy option for NT */ + break; + case (int) OPT_CONSOLE: + opt_console=1; + break; +#endif + case (int) OPT_FLUSH: + nisam_flush=myisam_flush=1; + flush_time=0; // No auto flush + break; + case OPT_LOW_PRIORITY_UPDATES: + thd_startup_options|=OPTION_LOW_PRIORITY_UPDATES; + low_priority_updates=1; + break; + case OPT_BOOTSTRAP: + opt_noacl=opt_bootstrap=1; + break; + case OPT_TABLE_TYPE: + { + int type; + if ((type=find_type(optarg, &ha_table_typelib, 2)) <= 0) + { + fprintf(stderr,"Unknown table type: %s\n",optarg); + exit(1); + } + default_table_type= (enum db_type) type; + break; + } + case OPT_DELAY_KEY_WRITE: + ha_open_options|=HA_OPEN_DELAY_KEY_WRITE; + myisam_delay_key_write=1; + break; + case OPT_SKIP_DELAY_KEY_WRITE: + myisam_delay_key_write=0; + break; + case 'C': + strmov(default_charset,optarg); + break; + case OPT_CHARSETS_DIR: + strmov(mysql_charsets_dir, optarg); + charsets_dir = mysql_charsets_dir; + break; +#include "sslopt-case.h" +#ifdef HAVE_BERKELEY_DB + case OPT_BDB_LOG: + berkeley_logdir=optarg; + break; + case OPT_BDB_HOME: + berkeley_home=optarg; + break; + case OPT_BDB_NOSYNC: + berkeley_init_flags|=DB_TXN_NOSYNC; + break; + case OPT_BDB_RECOVER: + berkeley_init_flags|=DB_RECOVER; + break; + case OPT_BDB_TMP: + berkeley_tmpdir=optarg; + break; + case OPT_BDB_LOCK: + { + int type; + if ((type=find_type(optarg, &berkeley_lock_typelib, 2)) > 0) + berkeley_lock_type=berkeley_lock_types[type-1]; + else + { + if (test_if_int(optarg,strlen(optarg))) + berkeley_lock_scan_time=atoi(optarg); + else + { + fprintf(stderr,"Unknown lock type: %s\n",optarg); + exit(1); + } + } + break; + } + case OPT_BDB_SKIP: + berkeley_skip=1; + break; +#endif + case OPT_MASTER_HOST: + master_host=optarg; + break; + case OPT_MASTER_USER: + master_user=optarg; + break; + case OPT_MASTER_PASSWORD: + master_password=optarg; + break; + case OPT_MASTER_INFO_FILE: + master_info_file=optarg; + break; + case OPT_MASTER_PORT: + master_port= atoi(optarg); + break; + case OPT_MASTER_CONNECT_RETRY: + master_connect_retry= atoi(optarg); + break; + + default: + fprintf(stderr,"%s: Unrecognized option: %c\n",my_progname,c); + usage(); + exit(1); + } + } + // Skipp empty arguments (from shell) + while (argc != optind && !argv[optind][0]) + optind++; + if (argc != optind) + { + fprintf(stderr,"%s: Too many parameters\n",my_progname); + usage(); + exit(1); + } + fix_paths(); + default_table_type_name=ha_table_typelib.type_names[default_table_type-1]; +} + + +#ifdef __WIN__ + +#ifndef KEY_SERVICE_PARAMETERS +#define KEY_SERVICE_PARAMETERS "SYSTEM\\CurrentControlSet\\Services\\MySql\\Parameters" +#endif + +#define COPY_KEY_VALUE(value) if (copy_key_value(hParametersKey,&(value),lpszValue)) return 1 +#define CHECK_KEY_TYPE(type,name) if ( type != dwKeyValueType ) { key_type_error(hParametersKey,name); return 1; } +#define SET_CHANGEABLE_VARVAL(varname) if (set_varval(hParametersKey,varname,szKeyValueName,dwKeyValueType,lpdwValue)) return 1; + +static void key_type_error(HKEY hParametersKey,const char *szKeyValueName) +{ + TCHAR szErrorMsg[512]; + RegCloseKey( hParametersKey ); + strxmov(szErrorMsg,TEXT("Value \""), + szKeyValueName, + TEXT("\" of registry key \"" KEY_SERVICE_PARAMETERS "\" has wrong type\n"),NullS); + fprintf(stderr, szErrorMsg); /* not unicode compatible */ +} + +static bool copy_key_value(HKEY hParametersKey, char **var, const char *value) +{ + if (!(*var=my_strdup(value,MYF(MY_WME)))) + { + RegCloseKey(hParametersKey); + fprintf(stderr, "Couldn't allocate memory for registry key value\n"); + return 1; + } + return 0; +} + +static bool set_varval(HKEY hParametersKey,const char *var, + const char *szKeyValueName, DWORD dwKeyValueType, + LPDWORD lpdwValue) +{ + CHECK_KEY_TYPE(dwKeyValueType, szKeyValueName ); + if (set_changeable_varval(var, *lpdwValue, changeable_vars)) + { + TCHAR szErrorMsg [ 512 ]; + RegCloseKey( hParametersKey ); + strxmov(szErrorMsg, + TEXT("Value \""), + szKeyValueName, + TEXT("\" of registry key \"" KEY_SERVICE_PARAMETERS "\" is invalid\n"),NullS); + fprintf( stderr, szErrorMsg ); /* not unicode compatible */ + return 1; + } + return 0; +} + + +static int get_service_parameters() +{ + DWORD dwLastError; + HKEY hParametersKey; + DWORD dwIndex; + TCHAR szKeyValueName [ 256 ]; + DWORD dwKeyValueName; + DWORD dwKeyValueType; + BYTE bKeyValueBuffer [ 512 ]; + DWORD dwKeyValueBuffer; + LPDWORD lpdwValue = (LPDWORD) &bKeyValueBuffer[0]; + LPCTSTR lpszValue = (LPCTSTR) &bKeyValueBuffer[0]; + + /* open parameters of service */ + dwLastError = (DWORD) RegOpenKeyEx( HKEY_LOCAL_MACHINE, + TEXT(KEY_SERVICE_PARAMETERS), 0, + KEY_READ, &hParametersKey ); + if ( dwLastError == ERROR_FILE_NOT_FOUND ) /* no parameters available */ + return 0; + if ( dwLastError != ERROR_SUCCESS ) + { + fprintf(stderr,"Can't open registry key \"" KEY_SERVICE_PARAMETERS "\" for reading\n" ); + return 1; + } + + /* enumerate all values of key */ + dwIndex = 0; + dwKeyValueName = sizeof( szKeyValueName ) / sizeof( TCHAR ); + dwKeyValueBuffer = sizeof( bKeyValueBuffer ); + while ( (dwLastError = (DWORD) RegEnumValue(hParametersKey, dwIndex, + szKeyValueName, &dwKeyValueName, + NULL, &dwKeyValueType, + &bKeyValueBuffer[0], + &dwKeyValueBuffer)) + != ERROR_NO_MORE_ITEMS ) + { + /* check if error occured */ + if ( dwLastError != ERROR_SUCCESS ) + { + RegCloseKey( hParametersKey ); + fprintf( stderr, "Can't enumerate values of registry key \"" KEY_SERVICE_PARAMETERS "\"\n" ); + return 1; + } + if ( lstrcmp(szKeyValueName, TEXT("BaseDir")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName); + strmov( mysql_home, lpszValue ); /* not unicode compatible */ + } + else if ( lstrcmp(szKeyValueName, TEXT("BindAddress")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName); + + my_bind_addr = (ulong) inet_addr( lpszValue ); + if ( my_bind_addr == (ulong) INADDR_NONE ) + { + struct hostent* ent; + + if ( !(*lpszValue) ) + { + char szHostName [ 256 ]; + if ( gethostname(szHostName, sizeof(szHostName)) == SOCKET_ERROR ) + { + RegCloseKey( hParametersKey ); + fprintf( stderr, "Can't get my own hostname\n" ); + return 1; + } + ent = gethostbyname( szHostName ); + } + else ent = gethostbyname( lpszValue ); + if ( !ent ) + { + RegCloseKey( hParametersKey ); + fprintf( stderr, "Can't resolve hostname!\n" ); + return 1; + } + my_bind_addr = (ulong) ((in_addr*)ent->h_addr_list[0])->s_addr; + } + } + else if ( lstrcmp(szKeyValueName, TEXT("BigTables")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName); + if ( *lpdwValue ) + thd_startup_options |= OPTION_BIG_TABLES; + else + thd_startup_options &= ~((ulong)OPTION_BIG_TABLES); + } + else if ( lstrcmp(szKeyValueName, TEXT("DataDir")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + strmov( mysql_real_data_home, lpszValue ); /* not unicode compatible */ + } + else if ( lstrcmp(szKeyValueName, TEXT("Locking")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + my_disable_locking = !(*lpdwValue); + } + else if ( lstrcmp(szKeyValueName, TEXT("LogFile")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + opt_log = 1; + COPY_KEY_VALUE( opt_logname ); + } + else if ( lstrcmp(szKeyValueName, TEXT("UpdateLogFile")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + opt_update_log = 1; + COPY_KEY_VALUE( opt_update_logname ); + } + else if ( lstrcmp(szKeyValueName, TEXT("BinaryLogFile")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + opt_bin_log = 1; + COPY_KEY_VALUE( opt_bin_logname ); + } + else if ( lstrcmp(szKeyValueName, TEXT("BinaryLogIndexFile")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + opt_bin_log = 1; + COPY_KEY_VALUE( opt_binlog_index_name ); + } + else if ( lstrcmp(szKeyValueName, TEXT("ISAMLogFile")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + COPY_KEY_VALUE( nisam_log_filename ); + (void) nisam_log( 1 ); + } + else if ( lstrcmp(szKeyValueName, TEXT("LongLogFormat")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + if ( *lpdwValue ) + opt_specialflag |= SPECIAL_LONG_LOG_FORMAT; + else + opt_specialflag &= ~((ulong)SPECIAL_LONG_LOG_FORMAT); + } + else if ( lstrcmp(szKeyValueName, TEXT("LowPriorityUpdates")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + if ( *lpdwValue ) + { + thd_startup_options |= OPTION_LOW_PRIORITY_UPDATES; + low_priority_updates = 1; + } + else + { + thd_startup_options &= ~((ulong)OPTION_LOW_PRIORITY_UPDATES); + low_priority_updates = 0; + } + } + else if ( lstrcmp(szKeyValueName, TEXT("Port")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + mysql_port = (unsigned int) *lpdwValue; + } + else if ( lstrcmp(szKeyValueName, TEXT("OldProtocol")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + protocol_version = *lpdwValue ? PROTOCOL_VERSION - 1 : PROTOCOL_VERSION; + } + else if ( lstrcmp(szKeyValueName, TEXT("HostnameResolving")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + if ( !*lpdwValue ) + opt_specialflag |= SPECIAL_NO_RESOLVE; + else + opt_specialflag &= ~((ulong)SPECIAL_NO_RESOLVE); + } + else if ( lstrcmp(szKeyValueName, TEXT("Networking")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + opt_disable_networking = !(*lpdwValue); + } + else if ( lstrcmp(szKeyValueName, TEXT("ShowDatabase")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + opt_disable_networking = !(*lpdwValue); + } + else if ( lstrcmp(szKeyValueName, TEXT("HostnameCaching")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + if ( !*lpdwValue ) + opt_specialflag |= SPECIAL_NO_HOST_CACHE; + else + opt_specialflag &= ~((ulong)SPECIAL_NO_HOST_CACHE); + } + else if ( lstrcmp(szKeyValueName, TEXT("ThreadPriority")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + if ( !(*lpdwValue) ) + opt_specialflag |= SPECIAL_NO_PRIOR; + else + opt_specialflag &= ~((ulong)SPECIAL_NO_PRIOR); + } + else if ( lstrcmp(szKeyValueName, TEXT("NamedPipe")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + COPY_KEY_VALUE( mysql_unix_port ); + } + else if ( lstrcmp(szKeyValueName, TEXT("TempDir")) == 0 ) + { + CHECK_KEY_TYPE( REG_SZ, szKeyValueName ); + COPY_KEY_VALUE( mysql_tmpdir ); + } + else if ( lstrcmp(szKeyValueName, TEXT("FlushTables")) == 0 ) + { + CHECK_KEY_TYPE( REG_DWORD, szKeyValueName ); + nisam_flush = myisam_flush= *lpdwValue ? 1 : 0; + } + else if ( lstrcmp(szKeyValueName, TEXT("BackLog")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "back_log" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("ConnectTimeout")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "connect_timeout" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("JoinBufferSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "join_buffer" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("KeyBufferSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "key_buffer" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("LongQueryTime")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "long_query_time" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxAllowedPacket")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "max_allowed_packet" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxConnections")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "max_connections" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxConnectErrors")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "max_connect_errors" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxInsertDelayedThreads")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "max_delayed_threads" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxJoinSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "max_join_size" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxSortLength")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "max_sort_length" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("NetBufferLength")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "net_buffer_length" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("RecordBufferSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "record_buffer" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("SortBufferSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "sort_buffer" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("TableCacheSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "table_cache" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("TmpTableSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "tmp_table_size" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("ThreadStackSize")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "thread_stack" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("WaitTimeout")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "wait_timeout" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("DelayedInsertTimeout")) + == 0 ) + { + SET_CHANGEABLE_VARVAL( "delayed_insert_timeout" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("DelayedInsertLimit")) == + 0 ) + { + SET_CHANGEABLE_VARVAL( "delayed_insert_limit" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("DelayedQueueSize")) == 0 + ) + { + SET_CHANGEABLE_VARVAL( "delayed_queue_size" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("FlushTime")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "flush_time" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("InteractiveTimeout")) == + 0 ) + { + SET_CHANGEABLE_VARVAL( "interactive_timeout" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("LowerCaseTableNames")) + == 0 ) + { + SET_CHANGEABLE_VARVAL( "lower_case_table_names" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxHeapTableSize")) == 0 + ) + { + SET_CHANGEABLE_VARVAL( "max_heap_table_size" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxTmpTables")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "max_tmp_tables" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("MaxWriteLockCount")) == + 0 ) + { + SET_CHANGEABLE_VARVAL( "max_write_lock_count" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("NetRetryCount")) == 0 ) + { + SET_CHANGEABLE_VARVAL( "net_retry_count" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("QueryBufferSize")) == 0 + ) + { + SET_CHANGEABLE_VARVAL( "query_buffer_size" ); + } + else if ( lstrcmp(szKeyValueName, TEXT("ThreadConcurrency")) == + 0 ) + { + SET_CHANGEABLE_VARVAL( "thread_concurrency" ); + } + else + { + TCHAR szErrorMsg [ 512 ]; + RegCloseKey( hParametersKey ); + lstrcpy( szErrorMsg, TEXT("Value \"") ); + lstrcat( szErrorMsg, szKeyValueName ); + lstrcat( szErrorMsg, TEXT("\" of registry key \"" KEY_SERVICE_PARAMETERS "\" is not defined by MySQL\n") ); + fprintf( stderr, szErrorMsg ); /* not unicode compatible */ + return 1; + } + + dwIndex++; + dwKeyValueName = sizeof( szKeyValueName ) / sizeof( TCHAR ); + dwKeyValueBuffer = sizeof( bKeyValueBuffer ); + } + RegCloseKey( hParametersKey ); + + /* paths are fixed by method get_options() */ + return 0; +} +#endif + + +static char *get_relative_path(const char *path) +{ + if (test_if_hard_path(path) && + is_prefix(path,DEFAULT_MYSQL_HOME) && + strcmp(DEFAULT_MYSQL_HOME,FN_ROOTDIR)) + { + path+=strlen(DEFAULT_MYSQL_HOME); + while (*path == FN_LIBCHAR) + path++; + } + return (char*) path; +} + + +static void fix_paths(void) +{ + (void) fn_format(mysql_home,mysql_home,"","",16); // Remove symlinks + convert_dirname(mysql_home); + convert_dirname(mysql_real_data_home); + convert_dirname(language); + (void) my_load_path(mysql_home,mysql_home,""); // Resolve current dir + (void) my_load_path(mysql_real_data_home,mysql_real_data_home,mysql_home); + (void) my_load_path(pidfile_name,pidfile_name,mysql_real_data_home); + + char buff[FN_REFLEN],*sharedir=get_relative_path(SHAREDIR); + if (test_if_hard_path(sharedir)) + strmov(buff,sharedir); /* purecov: tested */ + else + strxmov(buff,mysql_home,sharedir,NullS); + convert_dirname(buff); + (void) my_load_path(language,language,buff); + + /* If --character-sets-dir isn't given, use shared library dir */ + if (charsets_dir != mysql_charsets_dir) + { + strmov(strmov(mysql_charsets_dir,buff),CHARSET_DIR); + charsets_dir=mysql_charsets_dir; + } + + /* Add '/' to TMPDIR if needed */ + char *tmp= (char*) my_malloc(FN_REFLEN,MYF(MY_FAE)); + if (tmp) + { + strmov(tmp,mysql_tmpdir); + mysql_tmpdir=tmp; + convert_dirname(mysql_tmpdir); + mysql_tmpdir=(char*) my_realloc(mysql_tmpdir,strlen(mysql_tmpdir)+1, + MYF(MY_HOLD_ON_ERROR)); + } +} + + +#ifdef SET_RLIMIT_NOFILE +static uint set_maximum_open_files(uint max_file_limit) +{ + struct rlimit rlimit; + ulong old_cur; + + if (!getrlimit(RLIMIT_NOFILE,&rlimit)) + { + old_cur=rlimit.rlim_cur; + if (rlimit.rlim_cur >= max_file_limit) // Nothing to do + return rlimit.rlim_cur; /* purecov: inspected */ + rlimit.rlim_cur=rlimit.rlim_max=max_file_limit; + if (setrlimit(RLIMIT_NOFILE,&rlimit)) + { + sql_print_error("Warning: setrlimit couldn't increase number of open files to more than %ld", + old_cur); /* purecov: inspected */ + max_file_limit=old_cur; + } + else + { + (void) getrlimit(RLIMIT_NOFILE,&rlimit); + if ((uint) rlimit.rlim_cur != max_file_limit) + sql_print_error("Warning: setrlimit returned ok, but didn't change limits. Max open files is %ld", + (ulong) rlimit.rlim_cur); /* purecov: inspected */ + max_file_limit=rlimit.rlim_cur; + } + } + return max_file_limit; +} +#endif + + +/***************************************************************************** +** Instantiate templates +*****************************************************************************/ + +#ifdef __GNUC__ +/* Used templates */ +template class I_List<THD>; +template class I_List_iterator<THD>; +template class I_List<i_string>; +#endif |