lib_sql.cc 17.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * Copyright (c)  2000
 * SWsoft  company
 *
 * This material is provided "as is", with absolutely no warranty expressed
 * or implied. Any use is at your own risk.
 *
 * Permission to use or copy this software for any purpose is hereby granted 
 * without fee, provided the above notices are retained on all copies.
 * Permission to modify the code and to distribute modified code is granted,
 * provided the above notices are retained, and a notice that the code was
 * modified is included with the above copyright notice.
 *
unknown's avatar
unknown committed
14 15 16

  This code was modified by the MySQL team
*/
17

unknown's avatar
unknown committed
18 19 20 21
/*
  The following is needed to not cause conflicts when we include mysqld.cc
*/

22 23 24 25
#define main main1
#define mysql_unix_port mysql_inix_port1
#define mysql_port mysql_port1

26 27 28 29 30
extern "C"
{
  unsigned long max_allowed_packet, net_buffer_length;
}

unknown's avatar
unknown committed
31 32 33
#if defined (__WIN__)
#include "../sql/mysqld.cpp"
#else
34
#include "../sql/mysqld.cc"
unknown's avatar
unknown committed
35
#endif
36 37

#define SCRAMBLE_LENGTH 8
unknown's avatar
unknown committed
38
C_MODE_START
unknown's avatar
unknown committed
39
#include "lib_vio.c"
unknown's avatar
unknown committed
40

unknown's avatar
unknown committed
41 42 43 44 45
static int check_connections1(THD * thd);
static int check_connections2(THD * thd);
static bool check_user(THD *thd, enum_server_command command,
		       const char *user, const char *passwd, const char *db,
		       bool check_count);
unknown's avatar
unknown committed
46 47
char * get_mysql_home(){ return mysql_home;};
char * get_mysql_real_data_home(){ return mysql_real_data_home;};
48 49 50 51

bool lib_dispatch_command(enum enum_server_command command, NET *net,
			  const char *arg, ulong length)
{
52 53
  THD *thd=(THD *) net->vio->dest_thd;
  thd->store_globals();				// Fix if more than one connect
unknown's avatar
unknown committed
54 55
  thd->net.last_error[0]=0;			// Clear error message
  thd->net.last_errno=0;
unknown's avatar
unknown committed
56

57 58
  net_new_transaction(&thd->net);
  return dispatch_command(command, thd, (char *) arg, length + 1);
59 60 61
}


unknown's avatar
unknown committed
62
void lib_connection_phase(NET * net, int phase)
63
{
unknown's avatar
unknown committed
64 65 66 67 68 69 70 71 72 73 74
  THD * thd;
  thd = (THD *)(net->vio->dest_thd);
  if (thd)
  {
    switch (phase)
    {
    case 2: 
      check_connections2(thd);
      break;
    }
  }
75
}
unknown's avatar
unknown committed
76
C_MODE_END
unknown's avatar
unknown committed
77 78


79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
void start_embedded_conn1(NET * net)
{
  THD * thd = new THD;
  my_net_init(&thd->net,NULL);
  /* if (protocol_version>9) */
  thd->net.return_errno=1;
  thd->thread_id = thread_id++;

  Vio * v  = net->vio;
  if (!v)
  {
    v = vio_new(0,VIO_CLOSED,0);
    net->vio = v;
  }
  if (v)
  {
    v -> dest_thd = thd;
  } 
  thd->net.vio = v;
  if (thd->store_globals())
  {
100
    fprintf(stderr,"store_globals failed.\n");
101 102 103 104 105 106 107
    return;
  }

  thd->mysys_var=my_thread_var;
  thd->dbug_thread_id=my_thread_id();
  thd->thread_stack= (char*) &thd;

unknown's avatar
unknown committed
108
  if (thd->variables.max_join_size == (ulong) HA_POS_ERROR)
109 110 111 112 113 114
    thd->options |= OPTION_BIG_SELECTS;

  thd->proc_info=0;				// Remove 'login'
  thd->command=COM_SLEEP;
  thd->version=refresh_version;
  thd->set_time();
unknown's avatar
unknown committed
115
  bzero(thd->scramble, sizeof(thd->scramble));
116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149
  init_sql_alloc(&thd->mem_root,8192,8192);

  check_connections1(thd);
}




static int
check_connections1(THD *thd)
{
  uint connect_errors=0;
  NET *net= &thd->net;
  /*
  ** store the connection details
  */
  DBUG_PRINT("info", (("check_connections called by thread %d"),
	     thd->thread_id));
  DBUG_PRINT("general",("New connection received on %s",
			vio_description(net->vio)));
  if (!thd->host)                           // If TCP/IP connection
  {
      thd->host=(char*) localhost;
  }
  else /* Hostname given means that the connection was on a socket */
  {
    DBUG_PRINT("general",("Host: %s",thd->host));
    thd->ip=0;
    bzero((char*) &thd->remote,sizeof(struct sockaddr));
  }
  //vio_keepalive(net->vio, TRUE);

  /* nasty, but any other way? */
  uint pkt_len = 0;
unknown's avatar
unknown committed
150

151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
    char buff[80],*end;
    int client_flags = CLIENT_LONG_FLAG | CLIENT_CONNECT_WITH_DB |
	               CLIENT_TRANSACTIONS;
    LINT_INIT(pkt_len);

    end=strmov(buff,server_version)+1;
    int4store((uchar*) end,thd->thread_id);
    end+=4;
    memcpy(end,thd->scramble,SCRAMBLE_LENGTH+1);
    end+=SCRAMBLE_LENGTH +1;
    int2store(end,client_flags);
    end[2]=MY_CHARSET_CURRENT;

#define MIN_HANDSHAKE_SIZE	6

    int2store(end+3,thd->server_status);
    bzero(end+5,13);
    end+=18;
    if (net_write_command(net,protocol_version, buff,
			  (uint) (end-buff))) 
    {
      inc_host_errors(&thd->remote.sin_addr);
      return(ER_HANDSHAKE_ERROR);
    }
   return 0; 
}

static int
check_connections2(THD * thd)
{
unknown's avatar
unknown committed
181 182 183 184
  uint connect_errors=0;
  uint pkt_len = 0;
  NET * net = &thd -> net;
  if (protocol_version>9) net -> return_errno=1;
185

unknown's avatar
unknown committed
186 187 188 189 190 191
  if ( (pkt_len=my_net_read(net)) == packet_error ||
       pkt_len < MIN_HANDSHAKE_SIZE)
  {
    inc_host_errors(&thd->remote.sin_addr);
    return(ER_HANDSHAKE_ERROR);
  }
192 193 194 195 196 197

#ifdef _CUSTOMCONFIG_
#include "_cust_sql_parse.h"
#endif
  if (connect_errors)
    reset_host_errors(&thd->remote.sin_addr);
unknown's avatar
unknown committed
198
  if (thd->packet.alloc(thd->variables.net_buffer_length))
199 200 201 202
    return(ER_OUT_OF_RESOURCES);

  thd->client_capabilities=uint2korr(net->read_pos);

unknown's avatar
unknown committed
203
  thd->max_client_packet_length=uint3korr(net->read_pos+2);
204 205 206 207 208 209 210 211 212
  char *user=   (char*) net->read_pos+5;
  char *passwd= strend(user)+1;
  char *db=0;
  if (passwd[0] && strlen(passwd) != SCRAMBLE_LENGTH)
    return ER_HANDSHAKE_ERROR;
  if (thd->client_capabilities & CLIENT_CONNECT_WITH_DB)
    db=strend(passwd)+1;
  if (thd->client_capabilities & CLIENT_TRANSACTIONS)
    thd->net.return_status= &thd->server_status;
unknown's avatar
unknown committed
213
  net->read_timeout=thd->variables.net_read_timeout;
214 215 216 217 218 219 220 221 222 223
  if (check_user(thd,COM_CONNECT, user, passwd, db, 1))
    return (-1);
  thd->password=test(passwd[0]);
  return 0;
}

static bool check_user(THD *thd,enum_server_command command, const char *user,
		       const char *passwd, const char *db, bool check_count)
{
  NET *net= &thd->net;
224
  USER_RESOURCES ur;
225 226 227 228 229 230 231
  thd->db=0;

  if (!(thd->user = my_strdup(user, MYF(0))))
  {
    send_error(net,ER_OUT_OF_RESOURCES);
    return 1;
  }
unknown's avatar
unknown committed
232
  strmake(thd->priv_host, LOCAL_HOST, sizeof(thd->priv_host)-1);
233
  thd->master_access=acl_getroot(thd, thd->host, thd->ip, thd->user,
unknown's avatar
unknown committed
234
				 passwd, thd->scramble,
235
                                 &thd->priv_user, thd->priv_host,
236 237
				 protocol_version == 9 ||
				 !(thd->client_capabilities &
238 239
				   CLIENT_LONG_PASSWORD),&ur);
  DBUG_PRINT("info",
240
	     ("Capabilities: %d  packet_length: %d  Host: '%s'  User: '%s'  Using password: %s  Access: %u  db: '%s'",
unknown's avatar
unknown committed
241
	      thd->client_capabilities, thd->max_client_packet_length,
242
	      thd->host_or_ip, thd->priv_user,
243 244 245 246 247 248
	      passwd[0] ? "yes": "no",
	      thd->master_access, thd->db ? thd->db : "*none*"));
  if (thd->master_access & NO_ACCESS)
  {
    net_printf(net, ER_ACCESS_DENIED_ERROR,
	       thd->user,
249
	       thd->host_or_ip,
250 251 252
	       passwd[0] ? ER(ER_YES) : ER(ER_NO));
    mysql_log.write(thd,COM_CONNECT,ER(ER_ACCESS_DENIED_ERROR),
		    thd->user,
253
		    thd->host_or_ip,
254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273
		    passwd[0] ? ER(ER_YES) : ER(ER_NO));
    return(1);					// Error already given
  }
  if (check_count)
  {
    VOID(pthread_mutex_lock(&LOCK_thread_count));
    bool tmp=(thread_count - delayed_insert_threads >= max_connections &&
	      !(thd->master_access & PROCESS_ACL));
    VOID(pthread_mutex_unlock(&LOCK_thread_count));
    if (tmp)
    {						// Too many connections
      send_error(net, ER_CON_COUNT_ERROR);
      return(1);
    }
  }
  mysql_log.write(thd,command,
		  (thd->priv_user == thd->user ?
		   (char*) "%s@%s on %s" :
		   (char*) "%s@%s as anonymous on %s"),
		  user,
274
		  thd->host_or_ip,
275 276 277 278 279 280 281 282 283 284
		  db ? db : (char*) "");
  thd->db_access=0;
  if (db && db[0])
    return test(mysql_change_db(thd,db));
  else
    send_ok(net);				// Ready to handle questions
  return 0;					// ok
}


unknown's avatar
unknown committed
285 286 287 288 289 290
void THD::clear_error()
{
  net.last_error[0]= 0;
  net.last_errno= 0;
}

291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317
/*
  Make a copy of array and the strings array points to
*/

char **copy_arguments(int argc, char **argv)
{
  uint length= 0;
  char **from, **res, **end= argv+argc;

  for (from=argv ; from != end ; from++)
    length+= strlen(*from);

  if ((res= (char**) my_malloc(sizeof(argv)*(argc+1)+length+argc,
			       MYF(MY_WME))))
  {
    char **to= res, *to_str= (char*) (res+argc+1);
    for (from=argv ; from != end ;)
    {
      *to++= to_str;
      to_str= strmov(to_str, *from++)+1;
    }
    *to= 0;					// Last ptr should be null
  }
  return res;
}


unknown's avatar
unknown committed
318 319 320
extern "C"
{

unknown's avatar
unknown committed
321 322
static my_bool  org_my_init_done;
my_bool         server_inited;
323
char **		copy_arguments_ptr= 0; 
unknown's avatar
unknown committed
324

325
int STDCALL mysql_server_init(int argc, char **argv, char **groups)
326
{
unknown's avatar
unknown committed
327
  char glob_hostname[FN_REFLEN];
328

unknown's avatar
unknown committed
329 330 331 332 333 334
  /* This mess is to allow people to call the init function without
   * having to mess with a fake argv */
  int *argcp;
  char ***argvp;
  int fake_argc = 1;
  char *fake_argv[] = { (char *)"", 0 };
335
  const char *fake_groups[] = { "server", "embedded", 0 };
unknown's avatar
unknown committed
336 337 338
  if (argc)
  {
    argcp = &argc;
unknown's avatar
unknown committed
339
    argvp = (char***) &argv;
unknown's avatar
unknown committed
340 341 342 343
  }
  else
  {
    argcp = &fake_argc;
unknown's avatar
unknown committed
344
    argvp = (char ***) &fake_argv;
unknown's avatar
unknown committed
345 346
  }
  if (!groups)
347
    groups = (char**) fake_groups;
unknown's avatar
unknown committed
348

349 350
  my_umask=0660;		// Default umask for new files
  my_umask_dir=0700;		// Default umask for new directories
unknown's avatar
unknown committed
351 352

  /* Only call MY_INIT() if it hasn't been called before */
unknown's avatar
unknown committed
353
  if (!server_inited)
unknown's avatar
unknown committed
354
  {
unknown's avatar
unknown committed
355
    server_inited=1;
unknown's avatar
unknown committed
356 357 358
    org_my_init_done=my_init_done;
  }
  if (!org_my_init_done)
359
  {
unknown's avatar
unknown committed
360
    MY_INIT((char *)"mysql_embedded");	// init my_sys library & pthreads
361
  }
unknown's avatar
unknown committed
362

363 364 365 366 367 368 369 370
  /*
    Make a copy of the arguments to guard against applications that
    may change or move the initial arguments.
  */
  if (argvp == &argv)
    if (!(copy_arguments_ptr= argv= copy_arguments(argc, argv)))
      return 1;

371 372 373 374 375 376 377 378
  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);
unknown's avatar
unknown committed
379
    strmov(time_zone,tzname[tm_tmp.tm_isdst != 0 ? 1 : 0]);
380 381 382 383 384
  }
#else
  {
    struct tm *start_tm;
    start_tm=localtime(&start_time);
unknown's avatar
unknown committed
385
    strmov(time_zone,tzname[start_tm->tm_isdst != 0 ? 1 : 0]);
386 387 388 389
  }
#endif
#endif

unknown's avatar
unknown committed
390 391
  if (gethostname(glob_hostname,sizeof(glob_hostname)-4) < 0)
    strmov(glob_hostname,"mysql");
392
  load_defaults("my", (const char **) groups, argcp, argvp);
unknown's avatar
unknown committed
393
  defaults_argv=*argvp;
unknown's avatar
unknown committed
394 395 396

  /* Get default temporary directory */
  opt_mysql_tmpdir=getenv("TMPDIR");	/* Use this if possible */
unknown's avatar
unknown committed
397
#if defined( __WIN__) || defined(OS2)
unknown's avatar
unknown committed
398 399 400 401
  if (!opt_mysql_tmpdir)
    opt_mysql_tmpdir=getenv("TEMP");
  if (!opt_mysql_tmpdir)
    opt_mysql_tmpdir=getenv("TMP");
402
#endif
unknown's avatar
unknown committed
403 404 405
  if (!opt_mysql_tmpdir || !opt_mysql_tmpdir[0])
    opt_mysql_tmpdir=(char*) P_tmpdir;		/* purecov: inspected */

406
  set_options();
unknown's avatar
unknown committed
407
  get_options(*argcp, *argvp);
408
  set_server_version();
409 410 411 412

  DBUG_PRINT("info",("%s  Ver %s for %s on %s\n",my_progname,
		     server_version, SYSTEM_TYPE,MACHINE_TYPE));

413 414 415 416 417 418 419 420 421 422 423 424 425 426 427
  if (opt_error_log)
  {
    if (!log_error_file_ptr[0])
      fn_format(log_error_file, glob_hostname, mysql_data_home, ".err", 0);
    else
      fn_format(log_error_file, log_error_file_ptr, mysql_data_home, ".err",
		MY_UNPACK_FILENAME | MY_SAFE_PATH);
    if (!log_error_file[0])
      opt_error_log= 1;				// Too long file name
    else
    {
      freopen(log_error_file, "a+", stderr);
    }
  }

428 429
  /* These must be set early */

unknown's avatar
unknown committed
430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445
  (void) pthread_mutex_init(&LOCK_mysql_create_db,MY_MUTEX_INIT_SLOW);
  (void) pthread_mutex_init(&LOCK_Acl,MY_MUTEX_INIT_SLOW);
  (void) pthread_mutex_init(&LOCK_grant,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_open,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_thread_count,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_mapped_file,MY_MUTEX_INIT_SLOW);
  (void) pthread_mutex_init(&LOCK_status,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_error_log,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_delayed_insert,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_delayed_status,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_delayed_create,MY_MUTEX_INIT_SLOW);
  (void) pthread_mutex_init(&LOCK_manager,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_crypt,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_bytes_sent,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_bytes_received,MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_user_conn, MY_MUTEX_INIT_FAST);
unknown's avatar
unknown committed
446 447 448
  (void) pthread_mutex_init(&LOCK_rpl_status, MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_active_mi, MY_MUTEX_INIT_FAST);
  (void) pthread_mutex_init(&LOCK_global_system_variables, MY_MUTEX_INIT_FAST);
449 450 451 452 453
  (void) pthread_cond_init(&COND_thread_count,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_manager,NULL);
unknown's avatar
unknown committed
454
  (void) pthread_cond_init(&COND_rpl_status, NULL);
455

unknown's avatar
unknown committed
456
  if (set_default_charset_by_name(sys_charset.value, MYF(MY_WME)))
unknown's avatar
unknown committed
457 458 459 460
  {
    mysql_server_end();
    return 1;
  }
461 462 463 464 465 466 467 468 469
  charsets_list = list_charsets(MYF(MY_COMPILED_SETS|MY_CONFIG_SETS));

  /* 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);
  pthread_attr_setscope(&connection_attrib, PTHREAD_SCOPE_SYSTEM);

unknown's avatar
unknown committed
470 471
#if defined( SET_RLIMIT_NOFILE) || defined( OS2) 
  /* connections and databases needs lots of files */
472 473 474
  {
    uint wanted_files=10+(uint) max(max_connections*5,
				    max_connections+table_cache_size*2);
unknown's avatar
unknown committed
475 476
    set_if_bigger(wanted_files, open_files_limit);
    // Note that some system returns 0 if we succeed here:
477
    uint files=set_maximum_open_files(wanted_files);
unknown's avatar
unknown committed
478
    if (files && files < wanted_files && ! open_files_limit)
479 480 481 482 483 484 485 486 487 488 489 490 491 492
    {
      max_connections=	(ulong) min((files-10),max_connections);
      table_cache_size= (ulong) max((files-10-max_connections)/2,64);
      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();
unknown's avatar
unknown committed
493
  set_var_init();
494 495 496 497
  mysys_uses_curses=0;
#ifdef USE_REGEX
  regex_init();
#endif
498
  if (use_temp_pool && bitmap_init(&temp_pool,1024,1))
unknown's avatar
unknown committed
499 500 501 502
  {
    mysql_server_end();
    return 1;
  }
503 504

  /*
unknown's avatar
unknown committed
505
    We have enough space for fiddling with the argv, continue
506 507 508 509
  */
  umask(((~my_umask) & 0666));
  table_cache_init();
  hostname_cache_init();
unknown's avatar
unknown committed
510 511
  query_cache_result_size_limit(query_cache_limit);
  query_cache_resize(query_cache_size);
512 513 514
  randominit(&sql_rand,(ulong) start_time,(ulong) start_time/2);
  reset_floating_point_exceptions();
  init_thr_lock();
unknown's avatar
unknown committed
515
  init_slave_list();
516 517 518

  /* Setup log files */
  if (opt_log)
519
    open_log(&mysql_log, glob_hostname, opt_logname, ".log", NullS,
520
	     LOG_NORMAL, 0, 0, 0);
521
  if (opt_update_log)
unknown's avatar
unknown committed
522 523
  {
    open_log(&mysql_update_log, glob_hostname, opt_update_logname, "",
524
	     NullS, LOG_NEW, 0, 0, 0);
unknown's avatar
unknown committed
525 526
    using_update_log=1;
  }
unknown's avatar
unknown committed
527

528
  if (opt_slow_log)
unknown's avatar
unknown committed
529
    open_log(&mysql_slow_log, glob_hostname, opt_slow_logname, "-slow.log",
530
	     NullS, LOG_NORMAL, 0, 0, 0);
531 532 533 534 535
  if (ha_init())
  {
    sql_print_error("Can't init databases");
    exit(1);
  }
unknown's avatar
unknown committed
536
  ha_key_cache();
unknown's avatar
unknown committed
537
#if defined(HAVE_MLOCKALL) && defined(MCL_CURRENT)
538 539 540 541 542 543 544 545 546 547 548
  if (locked_in_memory && !geteuid())
  {
    if (mlockall(MCL_CURRENT))
    {
      sql_print_error("Warning: Failed to lock memory. Errno: %d\n",errno);
    }
    else
      locked_in_memory=1;
  }
#else
  locked_in_memory=0;
unknown's avatar
unknown committed
549
#endif
550 551 552

  if (opt_myisam_log)
    (void) mi_log( 1 );
unknown's avatar
unknown committed
553
  ft_init_stopwords();
554 555 556 557 558 559 560 561 562 563 564 565

  /*
    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);
  }
unknown's avatar
unknown committed
566
  opt_noacl = 1;				// No permissions
unknown's avatar
unknown committed
567
  if (acl_init((THD*) 0,opt_noacl))
568
  {
unknown's avatar
unknown committed
569 570
    mysql_server_end();
    return 1;
571
  }
unknown's avatar
unknown committed
572
  if (!opt_noacl)
unknown's avatar
unknown committed
573
    (void) grant_init((THD*) 0);
unknown's avatar
unknown committed
574
  init_max_user_conn();
575
  init_update_queries();
576 577 578 579 580 581

#ifdef HAVE_DLOPEN
  if (!opt_noacl)
    udf_init();
#endif

unknown's avatar
unknown committed
582 583 584 585 586 587 588 589 590 591 592 593 594
  if (opt_bin_log)
  {
    if (!opt_bin_logname)
    {
      char tmp[FN_REFLEN];
      /* TODO: The following should be using fn_format();  We just need to
	 first change fn_format() to cut the file name if it's too long.
      */
      strmake(tmp,glob_hostname,FN_REFLEN-5);
      strmov(strcend(tmp,'.'),"-bin");
      opt_bin_logname=my_strdup(tmp,MYF(MY_WME));
    }
    open_log(&mysql_bin_log, glob_hostname, opt_bin_logname, "-bin",
595
	     opt_binlog_index_name, LOG_BIN, 0, 0, max_binlog_size);
unknown's avatar
unknown committed
596 597 598
    using_update_log=1;
  }

599 600
  (void) thr_setconcurrency(concurrency);	// 10 by default

unknown's avatar
unknown committed
601 602 603 604 605
  if (
#ifdef HAVE_BERKELEY_DB
      !berkeley_skip ||
#endif
      (flush_time && flush_time != ~(ulong) 0L))
606 607 608 609 610
  {
    pthread_t hThread;
    if (pthread_create(&hThread,&connection_attrib,handle_manager,0))
      sql_print_error("Warning: Can't create thread to manage maintenance");
  }
unknown's avatar
unknown committed
611

612 613 614 615 616 617 618 619
  /*
    Update mysqld variables from client variables if set
    The client variables are set also by get_one_option() in mysqld.cc
  */
  if (max_allowed_packet)
    global_system_variables.max_allowed_packet= max_allowed_packet;
  if (net_buffer_length)
    global_system_variables.net_buffer_length= net_buffer_length;
unknown's avatar
unknown committed
620
  return 0;
621 622
}

unknown's avatar
unknown committed
623

624
void STDCALL mysql_server_end()
625
{
626 627
  my_free((char*) copy_arguments_ptr, MYF(MY_ALLOW_ZERO_PTR));
  copy_arguments_ptr=0;
unknown's avatar
unknown committed
628
  clean_up(0);
629 630 631
  /* If library called my_init(), free memory allocated by it */
  if (!org_my_init_done)
    my_end(0);
unknown's avatar
unknown committed
632 633
}

634
my_bool STDCALL mysql_thread_init()
unknown's avatar
unknown committed
635 636
{
#ifdef THREAD
unknown's avatar
unknown committed
637
  return my_thread_init();
unknown's avatar
unknown committed
638
#else
unknown's avatar
unknown committed
639
  return 0;
unknown's avatar
unknown committed
640 641
#endif
}
642

643
void STDCALL mysql_thread_end()
unknown's avatar
unknown committed
644 645
{
#ifdef THREAD
unknown's avatar
unknown committed
646
  my_thread_end();
unknown's avatar
unknown committed
647 648 649 650 651
#endif
}

void start_embedded_connection(NET * net)
{
unknown's avatar
unknown committed
652
  start_embedded_conn1(net);
653
}
unknown's avatar
unknown committed
654

unknown's avatar
unknown committed
655
void end_embedded_connection(NET * net)
656
{
unknown's avatar
unknown committed
657 658
  THD *thd = (THD *) net->vio->dest_thd;
  delete thd;
659
}
unknown's avatar
unknown committed
660 661

} /* extern "C" */