939
1125
if (!session || MyFlags & ME_NOREFRESH)
940
errmsg_printf(ERRMSG_LVL_ERROR, "%s: %s",internal::my_progname,str);
1126
errmsg_printf(ERRMSG_LVL_ERROR, "%s: %s",my_progname,str); /* purecov: inspected */
944
1130
static const char *load_default_groups[]= {
945
1131
DRIZZLE_CONFIG_NAME, "server", 0, 0};
947
static int show_starttime(drizzle_show_var *var, char *buff)
949
var->type= SHOW_LONG;
951
*((long *)buff)= (long) (time(NULL) - server_start_time);
955
static int show_flushstatustime(drizzle_show_var *var, char *buff)
957
var->type= SHOW_LONG;
959
*((long *)buff)= (long) (time(NULL) - flush_status_time);
963
static st_show_var_func_container show_starttime_cont= { &show_starttime };
965
static st_show_var_func_container show_flushstatustime_cont= { &show_flushstatustime };
968
Variables shown by SHOW STATUS in alphabetical order
970
static drizzle_show_var com_status_vars[]= {
971
{"admin_commands", (char*) offsetof(system_status_var, com_other), SHOW_LONG_STATUS},
972
{"alter_db", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_ALTER_DB]), SHOW_LONG_STATUS},
973
{"alter_table", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_ALTER_TABLE]), SHOW_LONG_STATUS},
974
{"analyze", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_ANALYZE]), SHOW_LONG_STATUS},
975
{"begin", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_BEGIN]), SHOW_LONG_STATUS},
976
{"change_db", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_CHANGE_DB]), SHOW_LONG_STATUS},
977
{"check", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_CHECK]), SHOW_LONG_STATUS},
978
{"checksum", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_CHECKSUM]), SHOW_LONG_STATUS},
979
{"commit", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_COMMIT]), SHOW_LONG_STATUS},
980
{"create_db", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_CREATE_DB]), SHOW_LONG_STATUS},
981
{"create_index", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_CREATE_INDEX]), SHOW_LONG_STATUS},
982
{"create_table", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_CREATE_TABLE]), SHOW_LONG_STATUS},
983
{"delete", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_DELETE]), SHOW_LONG_STATUS},
984
{"drop_db", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_DROP_DB]), SHOW_LONG_STATUS},
985
{"drop_index", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_DROP_INDEX]), SHOW_LONG_STATUS},
986
{"drop_table", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_DROP_TABLE]), SHOW_LONG_STATUS},
987
{"empty_query", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_EMPTY_QUERY]), SHOW_LONG_STATUS},
988
{"flush", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_FLUSH]), SHOW_LONG_STATUS},
989
{"insert", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_INSERT]), SHOW_LONG_STATUS},
990
{"insert_select", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_INSERT_SELECT]), SHOW_LONG_STATUS},
991
{"kill", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_KILL]), SHOW_LONG_STATUS},
992
{"load", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_LOAD]), SHOW_LONG_STATUS},
993
{"release_savepoint", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_RELEASE_SAVEPOINT]), SHOW_LONG_STATUS},
994
{"rename_table", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_RENAME_TABLE]), SHOW_LONG_STATUS},
995
{"replace", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_REPLACE]), SHOW_LONG_STATUS},
996
{"replace_select", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_REPLACE_SELECT]), SHOW_LONG_STATUS},
997
{"rollback", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_ROLLBACK]), SHOW_LONG_STATUS},
998
{"rollback_to_savepoint",(char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_ROLLBACK_TO_SAVEPOINT]), SHOW_LONG_STATUS},
999
{"savepoint", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_SAVEPOINT]), SHOW_LONG_STATUS},
1000
{"select", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_SELECT]), SHOW_LONG_STATUS},
1001
{"set_option", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_SET_OPTION]), SHOW_LONG_STATUS},
1002
{"show_create_db", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_SHOW_CREATE_DB]), SHOW_LONG_STATUS},
1003
{"show_create_table", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_SHOW_CREATE]), SHOW_LONG_STATUS},
1004
{"show_errors", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_SHOW_ERRORS]), SHOW_LONG_STATUS},
1005
{"show_warnings", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_SHOW_WARNS]), SHOW_LONG_STATUS},
1006
{"truncate", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_TRUNCATE]), SHOW_LONG_STATUS},
1007
{"unlock_tables", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_UNLOCK_TABLES]), SHOW_LONG_STATUS},
1008
{"update", (char*) offsetof(system_status_var, com_stat[(uint32_t) SQLCOM_UPDATE]), SHOW_LONG_STATUS},
1009
{NULL, NULL, SHOW_LONGLONG}
1012
static drizzle_show_var status_vars[]= {
1013
{"Aborted_clients", (char*) &aborted_threads, SHOW_LONGLONG},
1014
{"Aborted_connects", (char*) &aborted_connects, SHOW_LONGLONG},
1015
{"Bytes_received", (char*) offsetof(system_status_var, bytes_received), SHOW_LONGLONG_STATUS},
1016
{"Bytes_sent", (char*) offsetof(system_status_var, bytes_sent), SHOW_LONGLONG_STATUS},
1017
{"Connections", (char*) &global_thread_id, SHOW_INT_NOFLUSH},
1018
{"Created_tmp_disk_tables", (char*) offsetof(system_status_var, created_tmp_disk_tables), SHOW_LONG_STATUS},
1019
{"Created_tmp_tables", (char*) offsetof(system_status_var, created_tmp_tables), SHOW_LONG_STATUS},
1020
{"Flush_commands", (char*) &refresh_version, SHOW_INT_NOFLUSH},
1021
{"Handler_commit", (char*) offsetof(system_status_var, ha_commit_count), SHOW_LONG_STATUS},
1022
{"Handler_delete", (char*) offsetof(system_status_var, ha_delete_count), SHOW_LONG_STATUS},
1023
{"Handler_prepare", (char*) offsetof(system_status_var, ha_prepare_count), SHOW_LONG_STATUS},
1024
{"Handler_read_first", (char*) offsetof(system_status_var, ha_read_first_count), SHOW_LONG_STATUS},
1025
{"Handler_read_key", (char*) offsetof(system_status_var, ha_read_key_count), SHOW_LONG_STATUS},
1026
{"Handler_read_next", (char*) offsetof(system_status_var, ha_read_next_count), SHOW_LONG_STATUS},
1027
{"Handler_read_prev", (char*) offsetof(system_status_var, ha_read_prev_count), SHOW_LONG_STATUS},
1028
{"Handler_read_rnd", (char*) offsetof(system_status_var, ha_read_rnd_count), SHOW_LONG_STATUS},
1029
{"Handler_read_rnd_next", (char*) offsetof(system_status_var, ha_read_rnd_next_count), SHOW_LONG_STATUS},
1030
{"Handler_rollback", (char*) offsetof(system_status_var, ha_rollback_count), SHOW_LONG_STATUS},
1031
{"Handler_savepoint", (char*) offsetof(system_status_var, ha_savepoint_count), SHOW_LONG_STATUS},
1032
{"Handler_savepoint_rollback",(char*) offsetof(system_status_var, ha_savepoint_rollback_count), SHOW_LONG_STATUS},
1033
{"Handler_update", (char*) offsetof(system_status_var, ha_update_count), SHOW_LONG_STATUS},
1034
{"Handler_write", (char*) offsetof(system_status_var, ha_write_count), SHOW_LONG_STATUS},
1035
{"Key_blocks_not_flushed", (char*) offsetof(KEY_CACHE, global_blocks_changed), SHOW_KEY_CACHE_LONG},
1036
{"Key_blocks_unused", (char*) offsetof(KEY_CACHE, blocks_unused), SHOW_KEY_CACHE_LONG},
1037
{"Key_blocks_used", (char*) offsetof(KEY_CACHE, blocks_used), SHOW_KEY_CACHE_LONG},
1038
{"Key_read_requests", (char*) offsetof(KEY_CACHE, global_cache_r_requests), SHOW_KEY_CACHE_LONGLONG},
1039
{"Key_reads", (char*) offsetof(KEY_CACHE, global_cache_read), SHOW_KEY_CACHE_LONGLONG},
1040
{"Key_write_requests", (char*) offsetof(KEY_CACHE, global_cache_w_requests), SHOW_KEY_CACHE_LONGLONG},
1041
{"Key_writes", (char*) offsetof(KEY_CACHE, global_cache_write), SHOW_KEY_CACHE_LONGLONG},
1042
{"Last_query_cost", (char*) offsetof(system_status_var, last_query_cost), SHOW_DOUBLE_STATUS},
1043
{"Max_used_connections", (char*) &max_used_connections, SHOW_INT},
1044
{"Questions", (char*) offsetof(system_status_var, questions), SHOW_LONG_STATUS},
1045
{"Select_full_join", (char*) offsetof(system_status_var, select_full_join_count), SHOW_LONG_STATUS},
1046
{"Select_full_range_join", (char*) offsetof(system_status_var, select_full_range_join_count), SHOW_LONG_STATUS},
1047
{"Select_range", (char*) offsetof(system_status_var, select_range_count), SHOW_LONG_STATUS},
1048
{"Select_range_check", (char*) offsetof(system_status_var, select_range_check_count), SHOW_LONG_STATUS},
1049
{"Select_scan", (char*) offsetof(system_status_var, select_scan_count), SHOW_LONG_STATUS},
1050
{"Slow_queries", (char*) offsetof(system_status_var, long_query_count), SHOW_LONG_STATUS},
1051
{"Sort_merge_passes", (char*) offsetof(system_status_var, filesort_merge_passes), SHOW_LONG_STATUS},
1052
{"Sort_range", (char*) offsetof(system_status_var, filesort_range_count), SHOW_LONG_STATUS},
1053
{"Sort_rows", (char*) offsetof(system_status_var, filesort_rows), SHOW_LONG_STATUS},
1054
{"Sort_scan", (char*) offsetof(system_status_var, filesort_scan_count), SHOW_LONG_STATUS},
1055
{"Table_locks_immediate", (char*) &locks_immediate, SHOW_INT},
1056
{"Table_locks_waited", (char*) &locks_waited, SHOW_INT},
1057
{"Threads_connected", (char*) &connection_count, SHOW_INT},
1058
{"Uptime", (char*) &show_starttime_cont, SHOW_FUNC},
1059
{"Uptime_since_flush_status",(char*) &show_flushstatustime_cont, SHOW_FUNC},
1133
SHOW_VAR com_status_vars[]= {
1134
{"admin_commands", (char*) offsetof(STATUS_VAR, com_other), SHOW_LONG_STATUS},
1135
{"alter_db", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_ALTER_DB]), SHOW_LONG_STATUS},
1136
{"alter_table", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_ALTER_TABLE]), SHOW_LONG_STATUS},
1137
{"analyze", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_ANALYZE]), SHOW_LONG_STATUS},
1138
{"begin", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_BEGIN]), SHOW_LONG_STATUS},
1139
{"change_db", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_CHANGE_DB]), SHOW_LONG_STATUS},
1140
{"check", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_CHECK]), SHOW_LONG_STATUS},
1141
{"checksum", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_CHECKSUM]), SHOW_LONG_STATUS},
1142
{"commit", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_COMMIT]), SHOW_LONG_STATUS},
1143
{"create_db", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_CREATE_DB]), SHOW_LONG_STATUS},
1144
{"create_index", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_CREATE_INDEX]), SHOW_LONG_STATUS},
1145
{"create_table", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_CREATE_TABLE]), SHOW_LONG_STATUS},
1146
{"delete", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_DELETE]), SHOW_LONG_STATUS},
1147
{"drop_db", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_DROP_DB]), SHOW_LONG_STATUS},
1148
{"drop_index", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_DROP_INDEX]), SHOW_LONG_STATUS},
1149
{"drop_table", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_DROP_TABLE]), SHOW_LONG_STATUS},
1150
{"empty_query", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_EMPTY_QUERY]), SHOW_LONG_STATUS},
1151
{"flush", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_FLUSH]), SHOW_LONG_STATUS},
1152
{"insert", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_INSERT]), SHOW_LONG_STATUS},
1153
{"insert_select", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_INSERT_SELECT]), SHOW_LONG_STATUS},
1154
{"kill", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_KILL]), SHOW_LONG_STATUS},
1155
{"load", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_LOAD]), SHOW_LONG_STATUS},
1156
{"optimize", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_OPTIMIZE]), SHOW_LONG_STATUS},
1157
{"release_savepoint", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_RELEASE_SAVEPOINT]), SHOW_LONG_STATUS},
1158
{"rename_table", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_RENAME_TABLE]), SHOW_LONG_STATUS},
1159
{"replace", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_REPLACE]), SHOW_LONG_STATUS},
1160
{"replace_select", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_REPLACE_SELECT]), SHOW_LONG_STATUS},
1161
{"rollback", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_ROLLBACK]), SHOW_LONG_STATUS},
1162
{"rollback_to_savepoint",(char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_ROLLBACK_TO_SAVEPOINT]), SHOW_LONG_STATUS},
1163
{"savepoint", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SAVEPOINT]), SHOW_LONG_STATUS},
1164
{"select", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SELECT]), SHOW_LONG_STATUS},
1165
{"set_option", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SET_OPTION]), SHOW_LONG_STATUS},
1166
{"show_create_db", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_CREATE_DB]), SHOW_LONG_STATUS},
1167
{"show_create_table", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_CREATE]), SHOW_LONG_STATUS},
1168
{"show_databases", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_DATABASES]), SHOW_LONG_STATUS},
1169
{"show_engine_status", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_ENGINE_STATUS]), SHOW_LONG_STATUS},
1170
{"show_errors", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_ERRORS]), SHOW_LONG_STATUS},
1171
{"show_fields", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_FIELDS]), SHOW_LONG_STATUS},
1172
{"show_keys", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_KEYS]), SHOW_LONG_STATUS},
1173
{"show_open_tables", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_OPEN_TABLES]), SHOW_LONG_STATUS},
1174
{"show_processlist", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_PROCESSLIST]), SHOW_LONG_STATUS},
1175
{"show_status", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_STATUS]), SHOW_LONG_STATUS},
1176
{"show_table_status", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_TABLE_STATUS]), SHOW_LONG_STATUS},
1177
{"show_tables", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_TABLES]), SHOW_LONG_STATUS},
1178
{"show_variables", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_VARIABLES]), SHOW_LONG_STATUS},
1179
{"show_warnings", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_SHOW_WARNS]), SHOW_LONG_STATUS},
1180
{"truncate", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_TRUNCATE]), SHOW_LONG_STATUS},
1181
{"unlock_tables", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_UNLOCK_TABLES]), SHOW_LONG_STATUS},
1182
{"update", (char*) offsetof(STATUS_VAR, com_stat[(uint32_t) SQLCOM_UPDATE]), SHOW_LONG_STATUS},
1060
1183
{NULL, NULL, SHOW_LONGLONG}
1307
1433
if (default_storage_engine_str)
1309
1435
const std::string name(default_storage_engine_str);
1310
plugin::StorageEngine *engine;
1436
StorageEngine *engine;
1312
engine= plugin::StorageEngine::findByName(name);
1438
engine= ha_resolve_by_name(0, name);
1313
1439
if (engine == NULL)
1315
errmsg_printf(ERRMSG_LVL_ERROR, _("Unknown/unsupported storage engine: %s"),
1316
default_storage_engine_str);
1319
global_system_variables.storage_engine= engine;
1441
errmsg_printf(ERRMSG_LVL_ERROR, _("Unknown/unsupported table type: %s"),
1442
default_storage_engine_str);
1445
if (!engine->is_enabled())
1447
errmsg_printf(ERRMSG_LVL_ERROR, _("Default storage engine (%s) is not available"),
1448
default_storage_engine_str);
1450
//assert(global_system_variables.storage_engine);
1455
Need to unlock as global_system_variables.storage_engine
1456
was acquired during plugin_init()
1458
global_system_variables.storage_engine= engine;
1322
if (plugin::XaResourceManager::recoverAllXids(0))
1324
1464
unireg_abort(1);
1467
#if defined(HAVE_MLOCKALL) && defined(MCL_CURRENT)
1468
if (locked_in_memory && !getuid())
1470
if (setreuid((uid_t)-1, 0) == -1)
1471
{ // this should never happen
1472
sql_perror("setreuid");
1475
if (mlockall(MCL_CURRENT))
1477
if (global_system_variables.log_warnings)
1478
errmsg_printf(ERRMSG_LVL_WARN, _("Failed to lock memory. Errno: %d\n"),errno);
1479
locked_in_memory= 0;
1482
set_user(drizzled_user, user_info);
1327
1488
init_update_queries();
1493
int main(int argc, char **argv)
1495
ListenHandler listen_handler;
1500
#if defined(ENABLE_NLS)
1501
# if defined(HAVE_LOCALE_H)
1502
setlocale(LC_ALL, "");
1504
bindtextdomain("drizzle", LOCALEDIR);
1505
textdomain("drizzle");
1508
MY_INIT(argv[0]); // init my_sys library & pthreads
1509
/* nothing should come before this line ^^^ */
1511
/* Set signal used to kill Drizzle */
1512
#if defined(SIGUSR2)
1513
thr_kill_signal= thd_lib_detected == THD_LIB_LT ? SIGINT : SIGUSR2;
1515
thr_kill_signal= SIGINT;
1518
if (init_common_variables(DRIZZLE_CONFIG_NAME,
1519
argc, argv, load_default_groups))
1520
unireg_abort(1); // Will do exit
1525
select_thread=pthread_self();
1526
select_thread_in_use=1;
1528
check_data_home(drizzle_real_data_home);
1529
if (chdir(drizzle_real_data_home) && !opt_help)
1530
unireg_abort(1); /* purecov: inspected */
1531
drizzle_data_home= drizzle_data_home_buff;
1532
drizzle_data_home[0]=FN_CURLIB; // all paths are relative from here
1533
drizzle_data_home[1]=0;
1534
drizzle_data_home_len= 2;
1536
if ((user_info= check_user(drizzled_user)))
1538
#if defined(HAVE_MLOCKALL) && defined(MCL_CURRENT)
1539
if (locked_in_memory) // getuid() == 0 here
1540
set_effective_user(user_info);
1543
set_user(drizzled_user, user_info);
1551
if (init_server_components())
1556
if (listen_handler.bindAll(my_bind_addr_str, drizzled_port_timeout))
1560
init signals & alarm
1561
After this we can't quit by a simple unireg_abort
1563
error_handler_hook= my_message_sql;
1565
if (drizzle_rm_tmp_tables(listen_handler) ||
1566
my_tz_init((Session *)0, default_tz_name))
1569
select_thread_in_use=0;
1570
(void) pthread_kill(signal_thread, SIGTERM);
1572
(void) unlink(pidfile_name); // Not needed anymore
1579
errmsg_printf(ERRMSG_LVL_INFO, _(ER(ER_STARTUP)), my_progname, VERSION,
1580
COMPILATION_COMMENT);
1583
/* Listen for new connections and start new session for each connection
1584
accepted. The listen.getProtocol() method will return NULL when the server
1585
should be shutdown. */
1586
while ((protocol= listen_handler.getProtocol()) != NULL)
1588
if (!(session= new Session(protocol)))
1594
create_new_thread(session);
1597
/* (void) pthread_attr_destroy(&connection_attrib); */
1600
(void) pthread_mutex_lock(&LOCK_thread_count);
1601
select_thread_in_use=0; // For close_connections
1602
(void) pthread_mutex_unlock(&LOCK_thread_count);
1603
(void) pthread_cond_broadcast(&COND_thread_count);
1605
/* Wait until cleanup is done */
1606
(void) pthread_mutex_lock(&LOCK_thread_count);
1607
while (!ready_to_exit)
1608
pthread_cond_wait(&COND_server_end,&LOCK_thread_count);
1609
(void) pthread_mutex_unlock(&LOCK_thread_count);
1613
my_end(opt_endinfo ? MY_CHECK_ERROR | MY_GIVE_INFO : 0);
1619
Create new thread to handle incoming connection.
1621
This function will create new thread to handle the incoming
1622
connection. If there are idle cached threads one will be used.
1623
'session' will be pushed into 'threads'.
1625
In single-threaded mode (\#define ONE_THREAD) connection will be
1626
handled inside this function.
1628
@param[in,out] session Thread handle of future thread.
1631
static void create_new_thread(Session *session)
1633
Scheduler &thread_scheduler= get_thread_scheduler();
1637
if (connection_count > max_used_connections)
1638
max_used_connections= connection_count;
1641
The initialization of thread_id is done in create_embedded_session() for
1642
the embedded library.
1643
TODO: refactor this to avoid code duplication there
1645
session->thread_id= session->variables.pseudo_thread_id= thread_id++;
1648
If we error on creation we drop the connection and delete the session.
1650
pthread_mutex_lock(&LOCK_thread_count);
1651
session_list.push_back(session);
1652
pthread_mutex_unlock(&LOCK_thread_count);
1653
if (thread_scheduler.add_connection(session))
1655
char error_message_buff[DRIZZLE_ERRMSG_SIZE];
1657
session->killed= Session::KILL_CONNECTION; // Safety
1659
statistic_increment(aborted_connects, &LOCK_status);
1661
/* Can't use my_error() since store_globals has not been called. */
1662
snprintf(error_message_buff, sizeof(error_message_buff), ER(ER_CANT_CREATE_THREAD), 1); /* TODO replace will better error message */
1663
session->protocol->sendError(ER_CANT_CREATE_THREAD, error_message_buff);
1664
unlink_session(session);
1333
1669
/****************************************************************************
1334
1670
Handle start options
1335
1671
******************************************************************************/
1724
2134
(char**) &my_thread_stack_size,
1725
2135
(char**) &my_thread_stack_size, 0, GET_SIZE,
1726
2136
REQUIRED_ARG,DEFAULT_THREAD_STACK,
1727
UINT32_C(1024*512), SIZE_MAX, 0, 1024, 0},
2137
UINT32_C(1024*128), SIZE_MAX, 0, 1024, 0},
1728
2138
{"tmp_table_size", OPT_TMP_TABLE_SIZE,
1729
2139
N_("If an internal in-memory temporary table exceeds this size, Drizzle will"
1730
2140
" automatically convert it to an on-disk MyISAM table."),
1731
2141
(char**) &global_system_variables.tmp_table_size,
1732
2142
(char**) &max_system_variables.tmp_table_size, 0, GET_ULL,
1733
2143
REQUIRED_ARG, 16*1024*1024L, 1024, MAX_MEM_TABLE_SIZE, 0, 1, 0},
2144
{"transaction_alloc_block_size", OPT_TRANS_ALLOC_BLOCK_SIZE,
2145
N_("Allocation block size for transactions to be stored in binary log"),
2146
(char**) &global_system_variables.trans_alloc_block_size,
2147
(char**) &max_system_variables.trans_alloc_block_size, 0, GET_UINT,
2148
REQUIRED_ARG, QUERY_ALLOC_BLOCK_SIZE, 1024, ULONG_MAX, 0, 1024, 0},
2149
{"transaction_prealloc_size", OPT_TRANS_PREALLOC_SIZE,
2150
N_("Persistent buffer for transactions to be stored in binary log"),
2151
(char**) &global_system_variables.trans_prealloc_size,
2152
(char**) &max_system_variables.trans_prealloc_size, 0, GET_UINT,
2153
REQUIRED_ARG, TRANS_ALLOC_PREALLOC_SIZE, 1024, ULONG_MAX, 0, 1024, 0},
2154
{"wait_timeout", OPT_WAIT_TIMEOUT,
2155
N_("The number of seconds the server waits for activity on a connection "
2156
"before closing it."),
2157
(char**) &global_system_variables.net_wait_timeout,
2158
(char**) &max_system_variables.net_wait_timeout, 0, GET_UINT,
2159
REQUIRED_ARG, NET_WAIT_TIMEOUT, 1, LONG_TIMEOUT,
1734
2161
{0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}
2164
static int show_starttime(SHOW_VAR *var, char *buff)
2166
var->type= SHOW_LONG;
2168
*((long *)buff)= (long) (time(NULL) - server_start_time);
2172
static st_show_var_func_container
2173
show_starttime_cont= { &show_starttime };
2175
static int show_flushstatustime(SHOW_VAR *var, char *buff)
2177
var->type= SHOW_LONG;
2179
*((long *)buff)= (long) (time(NULL) - flush_status_time);
2183
static st_show_var_func_container
2184
show_flushstatustime_cont= { &show_flushstatustime };
2186
static int show_open_tables(SHOW_VAR *var, char *buff)
2188
var->type= SHOW_LONG;
2190
*((long *)buff)= (long)cached_open_tables();
2194
static int show_table_definitions(SHOW_VAR *var, char *buff)
2196
var->type= SHOW_LONG;
2198
*((long *)buff)= (long)cached_table_definitions();
2202
static st_show_var_func_container
2203
show_open_tables_cont= { &show_open_tables };
2204
static st_show_var_func_container
2205
show_table_definitions_cont= { &show_table_definitions };
2208
Variables shown by SHOW STATUS in alphabetical order
2211
SHOW_VAR status_vars[]= {
2212
{"Aborted_clients", (char*) &aborted_threads, SHOW_LONGLONG},
2213
{"Aborted_connects", (char*) &aborted_connects, SHOW_LONGLONG},
2214
{"Bytes_received", (char*) offsetof(STATUS_VAR, bytes_received), SHOW_LONGLONG_STATUS},
2215
{"Bytes_sent", (char*) offsetof(STATUS_VAR, bytes_sent), SHOW_LONGLONG_STATUS},
2216
{"Com", (char*) com_status_vars, SHOW_ARRAY},
2217
{"Connections", (char*) &thread_id, SHOW_INT_NOFLUSH},
2218
{"Created_tmp_disk_tables", (char*) offsetof(STATUS_VAR, created_tmp_disk_tables), SHOW_LONG_STATUS},
2219
{"Created_tmp_files", (char*) &my_tmp_file_created,SHOW_INT},
2220
{"Created_tmp_tables", (char*) offsetof(STATUS_VAR, created_tmp_tables), SHOW_LONG_STATUS},
2221
{"Flush_commands", (char*) &refresh_version, SHOW_INT_NOFLUSH},
2222
{"Handler_commit", (char*) offsetof(STATUS_VAR, ha_commit_count), SHOW_LONG_STATUS},
2223
{"Handler_delete", (char*) offsetof(STATUS_VAR, ha_delete_count), SHOW_LONG_STATUS},
2224
{"Handler_prepare", (char*) offsetof(STATUS_VAR, ha_prepare_count), SHOW_LONG_STATUS},
2225
{"Handler_read_first", (char*) offsetof(STATUS_VAR, ha_read_first_count), SHOW_LONG_STATUS},
2226
{"Handler_read_key", (char*) offsetof(STATUS_VAR, ha_read_key_count), SHOW_LONG_STATUS},
2227
{"Handler_read_next", (char*) offsetof(STATUS_VAR, ha_read_next_count), SHOW_LONG_STATUS},
2228
{"Handler_read_prev", (char*) offsetof(STATUS_VAR, ha_read_prev_count), SHOW_LONG_STATUS},
2229
{"Handler_read_rnd", (char*) offsetof(STATUS_VAR, ha_read_rnd_count), SHOW_LONG_STATUS},
2230
{"Handler_read_rnd_next", (char*) offsetof(STATUS_VAR, ha_read_rnd_next_count), SHOW_LONG_STATUS},
2231
{"Handler_rollback", (char*) offsetof(STATUS_VAR, ha_rollback_count), SHOW_LONG_STATUS},
2232
{"Handler_savepoint", (char*) offsetof(STATUS_VAR, ha_savepoint_count), SHOW_LONG_STATUS},
2233
{"Handler_savepoint_rollback",(char*) offsetof(STATUS_VAR, ha_savepoint_rollback_count), SHOW_LONG_STATUS},
2234
{"Handler_update", (char*) offsetof(STATUS_VAR, ha_update_count), SHOW_LONG_STATUS},
2235
{"Handler_write", (char*) offsetof(STATUS_VAR, ha_write_count), SHOW_LONG_STATUS},
2236
{"Key_blocks_not_flushed", (char*) offsetof(KEY_CACHE, global_blocks_changed), SHOW_KEY_CACHE_LONG},
2237
{"Key_blocks_unused", (char*) offsetof(KEY_CACHE, blocks_unused), SHOW_KEY_CACHE_LONG},
2238
{"Key_blocks_used", (char*) offsetof(KEY_CACHE, blocks_used), SHOW_KEY_CACHE_LONG},
2239
{"Key_read_requests", (char*) offsetof(KEY_CACHE, global_cache_r_requests), SHOW_KEY_CACHE_LONGLONG},
2240
{"Key_reads", (char*) offsetof(KEY_CACHE, global_cache_read), SHOW_KEY_CACHE_LONGLONG},
2241
{"Key_write_requests", (char*) offsetof(KEY_CACHE, global_cache_w_requests), SHOW_KEY_CACHE_LONGLONG},
2242
{"Key_writes", (char*) offsetof(KEY_CACHE, global_cache_write), SHOW_KEY_CACHE_LONGLONG},
2243
{"Last_query_cost", (char*) offsetof(STATUS_VAR, last_query_cost), SHOW_DOUBLE_STATUS},
2244
{"Max_used_connections", (char*) &max_used_connections, SHOW_INT},
2245
{"Open_files", (char*) &my_file_opened, SHOW_INT_NOFLUSH},
2246
{"Open_streams", (char*) &my_stream_opened, SHOW_INT_NOFLUSH},
2247
{"Open_table_definitions", (char*) &show_table_definitions_cont, SHOW_FUNC},
2248
{"Open_tables", (char*) &show_open_tables_cont, SHOW_FUNC},
2249
{"Opened_files", (char*) &my_file_total_opened, SHOW_INT_NOFLUSH},
2250
{"Opened_tables", (char*) offsetof(STATUS_VAR, opened_tables), SHOW_LONG_STATUS},
2251
{"Opened_table_definitions", (char*) offsetof(STATUS_VAR, opened_shares), SHOW_LONG_STATUS},
2252
{"Questions", (char*) offsetof(STATUS_VAR, questions), SHOW_LONG_STATUS},
2253
{"Select_full_join", (char*) offsetof(STATUS_VAR, select_full_join_count), SHOW_LONG_STATUS},
2254
{"Select_full_range_join", (char*) offsetof(STATUS_VAR, select_full_range_join_count), SHOW_LONG_STATUS},
2255
{"Select_range", (char*) offsetof(STATUS_VAR, select_range_count), SHOW_LONG_STATUS},
2256
{"Select_range_check", (char*) offsetof(STATUS_VAR, select_range_check_count), SHOW_LONG_STATUS},
2257
{"Select_scan", (char*) offsetof(STATUS_VAR, select_scan_count), SHOW_LONG_STATUS},
2258
{"Slow_queries", (char*) offsetof(STATUS_VAR, long_query_count), SHOW_LONG_STATUS},
2259
{"Sort_merge_passes", (char*) offsetof(STATUS_VAR, filesort_merge_passes), SHOW_LONG_STATUS},
2260
{"Sort_range", (char*) offsetof(STATUS_VAR, filesort_range_count), SHOW_LONG_STATUS},
2261
{"Sort_rows", (char*) offsetof(STATUS_VAR, filesort_rows), SHOW_LONG_STATUS},
2262
{"Sort_scan", (char*) offsetof(STATUS_VAR, filesort_scan_count), SHOW_LONG_STATUS},
2263
{"Table_locks_immediate", (char*) &locks_immediate, SHOW_INT},
2264
{"Table_locks_waited", (char*) &locks_waited, SHOW_INT},
2265
{"Threads_connected", (char*) &connection_count, SHOW_INT},
2266
{"Uptime", (char*) &show_starttime_cont, SHOW_FUNC},
2267
{"Uptime_since_flush_status",(char*) &show_flushstatustime_cont, SHOW_FUNC},
2268
{NULL, NULL, SHOW_LONGLONG}
1737
2271
static void print_version(void)
1740
2274
Note: the instance manager keys off the string 'Ver' so it can find the
1741
2275
version from the output of 'drizzled --version', so don't change it!
1743
printf("%s Ver %s for %s-%s on %s (%s)\n",internal::my_progname,
1744
PANDORA_RELEASE_VERSION, HOST_VENDOR, HOST_OS, HOST_CPU,
1745
COMPILATION_COMMENT);
2277
printf("%s Ver %s for %s-%s on %s (%s)\n",my_progname,
2278
VERSION, HOST_VENDOR, HOST_OS, HOST_CPU, COMPILATION_COMMENT);
1748
2281
static void usage(void)
2064
2656
pos[0]= FN_LIBCHAR;
2067
internal::convert_dirname(drizzle_real_data_home,drizzle_real_data_home,NULL);
2068
(void) internal::fn_format(buff, drizzle_real_data_home, "", "",
2659
convert_dirname(drizzle_real_data_home,drizzle_real_data_home,NULL);
2660
(void) fn_format(buff, drizzle_real_data_home, "", "",
2069
2661
(MY_RETURN_REAL_PATH|MY_RESOLVE_SYMLINKS));
2070
(void) internal::unpack_dirname(drizzle_unpacked_real_data_home, buff);
2071
internal::convert_dirname(language,language,NULL);
2072
(void) internal::my_load_path(drizzle_home, drizzle_home,""); // Resolve current dir
2073
(void) internal::my_load_path(drizzle_real_data_home, drizzle_real_data_home,drizzle_home);
2074
(void) internal::my_load_path(pidfile_name, pidfile_name,drizzle_real_data_home);
2076
if (opt_plugin_dir_ptr == NULL)
2078
/* No plugin dir has been specified. Figure out where the plugins are */
2079
if (progname[0] != FN_LIBCHAR)
2081
/* We have a relative path and need to find the absolute */
2082
char working_dir[FN_REFLEN];
2083
char *working_dir_ptr= working_dir;
2084
working_dir_ptr= getcwd(working_dir_ptr, FN_REFLEN);
2085
string new_path(working_dir);
2086
if (*(new_path.end()-1) != '/')
2087
new_path.push_back('/');
2088
if (progname[0] == '.' && progname[1] == '/')
2089
new_path.append(progname.substr(2));
2091
new_path.append(progname);
2092
progname.swap(new_path);
2095
/* Now, trim off the exe name */
2096
string progdir(progname.substr(0, progname.rfind(FN_LIBCHAR)+1));
2097
if (progdir.rfind(".libs/") != string::npos)
2099
progdir.assign(progdir.substr(0, progdir.rfind(".libs/")));
2101
string testfile(progdir);
2102
testfile.append("drizzled.o");
2103
struct stat testfile_stat;
2104
if (stat(testfile.c_str(), &testfile_stat))
2106
/* drizzled.o doesn't exist - we are not in a source dir.
2109
(void) internal::my_load_path(opt_plugin_dir, get_relative_path(PKGPLUGINDIR),
2114
/* We are in a source dir! Plugin dir is ../plugin/.libs */
2115
size_t last_libchar_pos= progdir.rfind(FN_LIBCHAR,progdir.size()-2)+1;
2116
string source_plugindir(progdir.substr(0,last_libchar_pos));
2117
source_plugindir.append("plugin/.libs");
2118
(void) internal::my_load_path(opt_plugin_dir, source_plugindir.c_str(), "");
2123
(void) internal::my_load_path(opt_plugin_dir, opt_plugin_dir_ptr, drizzle_home);
2662
(void) unpack_dirname(drizzle_unpacked_real_data_home, buff);
2663
convert_dirname(language,language,NULL);
2664
(void) my_load_path(drizzle_home, drizzle_home,""); // Resolve current dir
2665
(void) my_load_path(drizzle_real_data_home, drizzle_real_data_home,drizzle_home);
2666
(void) my_load_path(pidfile_name, pidfile_name,drizzle_real_data_home);
2667
(void) my_load_path(opt_plugin_dir, opt_plugin_dir_ptr ? opt_plugin_dir_ptr :
2668
get_relative_path(PKGPLUGINDIR),
2125
2670
opt_plugin_dir_ptr= opt_plugin_dir;
2127
2672
const char *sharedir= get_relative_path(PKGDATADIR);
2128
if (internal::test_if_hard_path(sharedir))
2129
strncpy(buff,sharedir,sizeof(buff)-1);
2673
if (test_if_hard_path(sharedir))
2674
strncpy(buff,sharedir,sizeof(buff)-1); /* purecov: tested */
2132
2677
strcpy(buff, drizzle_home);
2133
2678
strncat(buff, sharedir, sizeof(buff)-strlen(drizzle_home)-1);
2135
internal::convert_dirname(buff,buff,NULL);
2136
(void) internal::my_load_path(language,language,buff);
2680
convert_dirname(buff,buff,NULL);
2681
(void) my_load_path(language,language,buff);
2139
2684
char *tmp_string;
2173
} /* namespace drizzled */
2175
using namespace drizzled;
2178
static void init_signals(void)
2181
struct sigaction sa;
2183
if (!(test_flags.test(TEST_NO_STACKTRACE) ||
2184
test_flags.test(TEST_CORE_ON_SIGNAL)))
2186
sa.sa_flags = SA_RESETHAND | SA_NODEFER;
2187
sigemptyset(&sa.sa_mask);
2188
sigprocmask(SIG_SETMASK,&sa.sa_mask,NULL);
2191
sa.sa_handler=handle_segfault;
2192
sigaction(SIGSEGV, &sa, NULL);
2193
sigaction(SIGABRT, &sa, NULL);
2195
sigaction(SIGBUS, &sa, NULL);
2197
sigaction(SIGILL, &sa, NULL);
2198
sigaction(SIGFPE, &sa, NULL);
2201
if (test_flags.test(TEST_CORE_ON_SIGNAL))
2203
/* Change limits so that we will get a core file */
2205
rl.rlim_cur = rl.rlim_max = RLIM_INFINITY;
2206
if (setrlimit(RLIMIT_CORE, &rl) && global_system_variables.log_warnings)
2207
errmsg_printf(ERRMSG_LVL_WARN,
2208
_("setrlimit could not change the size of core files "
2209
"to 'infinity'; We may not be able to generate a "
2210
"core file on signals"));
2212
(void) sigemptyset(&set);
2213
my_sigset(SIGPIPE,SIG_IGN);
2214
sigaddset(&set,SIGPIPE);
2215
#ifndef IGNORE_SIGHUP_SIGQUIT
2216
sigaddset(&set,SIGQUIT);
2217
sigaddset(&set,SIGHUP);
2219
sigaddset(&set,SIGTERM);
2221
/* Fix signals if blocked by parents (can happen on Mac OS X) */
2222
sigemptyset(&sa.sa_mask);
2224
sa.sa_handler = print_signal_warning;
2225
sigaction(SIGTERM, &sa, (struct sigaction*) 0);
2227
sa.sa_handler = print_signal_warning;
2228
sigaction(SIGHUP, &sa, (struct sigaction*) 0);
2230
sigaddset(&set,SIGTSTP);
2232
if (test_flags.test(TEST_SIGINT))
2234
my_sigset(thr_kill_signal, end_thread_signal);
2236
sigdelset(&set, thr_kill_signal);
2239
sigaddset(&set,SIGINT);
2240
sigprocmask(SIG_SETMASK,&set,NULL);
2241
pthread_sigmask(SIG_SETMASK,&set,NULL);
2245
int main(int argc, char **argv)
2247
#if defined(ENABLE_NLS)
2248
# if defined(HAVE_LOCALE_H)
2249
setlocale(LC_ALL, "");
2251
bindtextdomain("drizzle", LOCALEDIR);
2252
textdomain("drizzle");
2255
plugin::Registry &plugins= plugin::Registry::singleton();
2256
plugin::Client *client;
2259
MY_INIT(argv[0]); // init my_sys library & pthreads
2260
/* nothing should come before this line ^^^ */
2262
/* Set signal used to kill Drizzle */
2263
#if defined(SIGUSR2)
2264
thr_kill_signal= internal::thd_lib_detected == THD_LIB_LT ? SIGINT : SIGUSR2;
2266
thr_kill_signal= SIGINT;
2269
if (init_common_variables(DRIZZLE_CONFIG_NAME,
2270
argc, argv, load_default_groups))
2271
unireg_abort(1); // Will do exit
2276
select_thread=pthread_self();
2277
select_thread_in_use=1;
2279
if (chdir(drizzle_real_data_home) && !opt_help)
2281
errmsg_printf(ERRMSG_LVL_ERROR, _("Data directory %s does not exist\n"), drizzle_real_data_home);
2284
drizzle_data_home= drizzle_data_home_buff;
2285
drizzle_data_home[0]=FN_CURLIB; // all paths are relative from here
2286
drizzle_data_home[1]=0;
2287
drizzle_data_home_len= 2;
2289
if ((user_info= check_user(drizzled_user)))
2291
set_user(drizzled_user, user_info);
2299
if (init_server_components(plugins))
2302
if (plugin::Listen::setup())
2306
init signals & alarm
2307
After this we can't quit by a simple unireg_abort
2309
error_handler_hook= my_message_sql;
2311
assert(plugin::num_trx_monitored_objects > 0);
2312
if (drizzle_rm_tmp_tables() ||
2313
my_tz_init((Session *)0, default_tz_name))
2316
select_thread_in_use=0;
2317
(void) pthread_kill(signal_thread, SIGTERM);
2319
(void) unlink(pidfile_name); // Not needed anymore
2326
errmsg_printf(ERRMSG_LVL_INFO, _(ER(ER_STARTUP)), internal::my_progname,
2327
PANDORA_RELEASE_VERSION, COMPILATION_COMMENT);
2330
/* Listen for new connections and start new session for each connection
2331
accepted. The listen.getClient() method will return NULL when the server
2332
should be shutdown. */
2333
while ((client= plugin::Listen::getClient()) != NULL)
2335
if (!(session= new Session(client)))
2341
/* If we error on creation we drop the connection and delete the session. */
2342
if (session->schedule())
2343
Session::unlink(session);
2346
/* (void) pthread_attr_destroy(&connection_attrib); */
2349
(void) pthread_mutex_lock(&LOCK_thread_count);
2350
select_thread_in_use=0; // For close_connections
2351
(void) pthread_mutex_unlock(&LOCK_thread_count);
2352
(void) pthread_cond_broadcast(&COND_thread_count);
2354
/* Wait until cleanup is done */
2355
(void) pthread_mutex_lock(&LOCK_thread_count);
2356
while (!ready_to_exit)
2357
pthread_cond_wait(&COND_server_end,&LOCK_thread_count);
2358
(void) pthread_mutex_unlock(&LOCK_thread_count);
2361
plugin::Registry::shutdown();
2718
/*****************************************************************************
2719
Instantiate templates
2720
*****************************************************************************/
2722
#ifdef HAVE_EXPLICIT_TEMPLATE_INSTANTIATION
2723
/* Used templates */
2724
template class I_List<i_string>;
2725
template class I_List<i_string_pair>;
2726
template class I_List<Statement>;
2727
template class I_List_iterator<Statement>;