mirror of
https://github.com/MariaDB/server.git
synced 2025-01-26 00:34:18 +01:00
f3e4ce926d
into mysql.com:/home/kent/bk/main/mysql-5.1 BUILD/Makefile.am: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~10: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~11: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~12: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~13: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~14: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~15: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~1: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~2f6eabb2f69cb33d: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~2: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~3: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~4ef559bc8b4695f7: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~4: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~5: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~6: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~7: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~84669765249a4bad: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~8: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~9: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~c20dcd005f596740: Auto merged BitKeeper/deleted/.del-CMakeLists.txt~dd682cce1d53c0b4: Auto merged BitKeeper/deleted/.del-Makefile.am~2: Auto merged BitKeeper/deleted/.del-Makefile.am~ab5c84d46412dc2e: Auto merged BitKeeper/deleted/.del-Makefile.am~de166d6fcac3b9b6: Auto merged BitKeeper/deleted/.del-Makefile.am~e5b911533dad2713: Auto merged BitKeeper/deleted/.del-Makefile.am~ead19441cc5ff35c: Auto merged BitKeeper/deleted/.del-Makefile.am~f87185e232d7c4f: Auto merged BitKeeper/deleted/.del-Makefile.in: Auto merged BitKeeper/deleted/.del-ReadMe.txt~573b1e4ebab241e1: Auto merged BitKeeper/deleted/.del-build-vs71.bat: Auto merged BitKeeper/deleted/.del-build-vs8.bat: Auto merged BitKeeper/deleted/.del-configure.js: Auto merged BitKeeper/deleted/.del-copy_mysql_files.bat~f6878eeb80173de9: Auto merged BitKeeper/deleted/.del-ha_berkeley.cc: Auto merged BitKeeper/deleted/.del-ha_berkeley.h: Auto merged BitKeeper/deleted/.del-make_win_bin_dist: Auto merged BitKeeper/deleted/.del-make_win_src_distribution.sh~f80d8fca44e4e5f1: Auto merged BitKeeper/deleted/.del-my_create_tables.c~c121a0c4c427ebb: Auto merged BitKeeper/deleted/.del-mysql_explain_log.sh~5ddc62808e16bd57: Auto merged BitKeeper/deleted/.del-mysql_thr.c~20772782813d1274: Auto merged BitKeeper/deleted/.del-mysql_upgrade.sh~826da969ccf96ef: Auto merged BitKeeper/deleted/.del-mysqlmanager.c~e97636d71145a0b: Auto merged BitKeeper/deleted/.del-prepare~773a10a535120a7e: Auto merged BitKeeper/deleted/.del-print-limit-table~b8e808031daa3758: Auto merged BitKeeper/deleted/.del-sql_manager.h: Auto merged BitKeeper/deleted/.del-thr_test.c~70fc0971c72f2a95: Auto merged Docs/Makefile.am: Auto merged Docs/generate-text-files.pl: Auto merged client/Makefile.am: Auto merged client/client_priv.h: Auto merged client/mysqladmin.cc: Auto merged client/mysqlimport.c: Auto merged client/mysqlshow.c: Auto merged dbug/Makefile.am: Auto merged extra/Makefile.am: Auto merged extra/yassl/taocrypt/benchmark/Makefile.am: Auto merged extra/yassl/taocrypt/test/Makefile.am: Auto merged include/Makefile.am: Auto merged include/my_time.h: Auto merged libmysql/Makefile.am: Auto merged libmysql_r/Makefile.am: Auto merged libmysqld/Makefile.am: Auto merged libmysqld/embedded_priv.h: Auto merged mysql-test/Makefile.am: Auto merged mysql-test/install_test_db.sh: Auto merged mysql-test/lib/mtr_cases.pl: Auto merged mysql-test/lib/mtr_io.pl: Auto merged mysql-test/lib/mtr_match.pl: Auto merged mysql-test/lib/mtr_misc.pl: Auto merged mysql-test/lib/mtr_process.pl: Auto merged mysql-test/lib/mtr_report.pl: Auto merged mysql-test/lib/mtr_timer.pl: Auto merged mysql-test/ndb/ndbcluster.sh: Auto merged mysys/Makefile.am: Auto merged mysys/my_gethostbyname.c: Auto merged mysys/my_getopt.c: Auto merged mysys/my_handler.c: Auto merged regex/Makefile.am: Auto merged scripts/Makefile.am: Auto merged scripts/fill_func_tables.sh: Auto merged scripts/make_binary_distribution.sh: Auto merged scripts/mysql_convert_table_format.sh: Auto merged scripts/mysql_create_system_tables.sh: Auto merged scripts/mysql_fix_privilege_tables.sh: Auto merged scripts/mysql_install_db.sh: Auto merged server-tools/instance-manager/IMService.cpp: Auto merged server-tools/instance-manager/WindowsService.cpp: Auto merged server-tools/instance-manager/listener.cc: Auto merged server-tools/instance-manager/log.cc: Auto merged server-tools/instance-manager/log.h: Auto merged server-tools/instance-manager/manager.cc: Auto merged server-tools/instance-manager/messages.cc: Auto merged server-tools/instance-manager/mysql_connection.cc: Auto merged server-tools/instance-manager/mysqlmanager.cc: Auto merged server-tools/instance-manager/options.cc: Auto merged server-tools/instance-manager/options.h: Auto merged server-tools/instance-manager/portability.h: Auto merged server-tools/instance-manager/priv.cc: Auto merged server-tools/instance-manager/protocol.cc: Auto merged server-tools/instance-manager/protocol.h: Auto merged server-tools/instance-manager/thread_registry.cc: Auto merged server-tools/instance-manager/thread_registry.h: Auto merged server-tools/instance-manager/user_map.cc: Auto merged server-tools/instance-manager/user_map.h: Auto merged sql/Makefile.am: Auto merged sql/discover.cc: Auto merged sql/field.cc: Auto merged sql/field.h: Auto merged sql/filesort.cc: Auto merged sql/gen_lex_hash.cc: Auto merged sql/handler.cc: Auto merged sql/handler.h: Auto merged sql/hostname.cc: Auto merged sql/init.cc: Auto merged sql/item.cc: Auto merged sql/item.h: Auto merged sql/item_cmpfunc.cc: Auto merged sql/item_create.h: Auto merged sql/item_func.h: Auto merged sql/item_geofunc.cc: Auto merged sql/item_strfunc.cc: Auto merged sql/item_sum.h: Auto merged sql/item_timefunc.h: Auto merged sql/item_uniq.cc: Auto merged sql/key.cc: Auto merged sql/lex_symbol.h: Auto merged sql/lock.cc: Auto merged sql/log_event.h: Auto merged sql/my_decimal.cc: Auto merged sql/my_decimal.h: Auto merged sql/my_lock.c: Auto merged sql/opt_range.cc: Auto merged sql/opt_range.h: Auto merged sql/password.c: Auto merged sql/procedure.h: Auto merged sql/protocol.h: Auto merged sql/records.cc: Auto merged sql/repl_failsafe.cc: Auto merged sql/set_var.h: Auto merged sql/spatial.h: Auto merged sql/sql_acl.h: Auto merged sql/sql_analyse.cc: Auto merged sql/sql_analyse.h: Auto merged sql/sql_base.cc: Auto merged sql/sql_cache.h: Auto merged sql/sql_class.cc: Auto merged sql/sql_class.h: Auto merged sql/sql_crypt.cc: Auto merged sql/sql_cursor.cc: Auto merged sql/sql_do.cc: Auto merged sql/sql_insert.cc: Auto merged sql/sql_lex.cc: Auto merged sql/sql_lex.h: Auto merged sql/sql_load.cc: Auto merged sql/sql_manager.cc: Auto merged sql/sql_map.cc: Auto merged sql/sql_olap.cc: Auto merged sql/sql_parse.cc: Auto merged sql/sql_rename.cc: Auto merged sql/sql_repl.cc: Auto merged sql/sql_repl.h: Auto merged sql-bench/Makefile.am: Auto merged sql-bench/as3ap.sh: Auto merged sql-bench/bench-count-distinct.sh: Auto merged sql-bench/bench-init.pl.sh: Auto merged sql-bench/compare-results.sh: Auto merged sql-bench/copy-db.sh: Auto merged sql-bench/crash-me.sh: Auto merged sql-bench/run-all-tests.sh: Auto merged sql-bench/server-cfg.sh: Auto merged sql-bench/test-ATIS.sh: Auto merged sql-bench/test-alter-table.sh: Auto merged sql-bench/test-big-tables.sh: Auto merged sql-bench/test-connect.sh: Auto merged sql-bench/test-create.sh: Auto merged sql-bench/test-insert.sh: Auto merged sql-bench/test-select.sh: Auto merged sql-bench/test-transactions.sh: Auto merged sql-bench/test-wisconsin.sh: Auto merged sql-common/my_time.c: Auto merged sql/sql_select.cc: Auto merged sql/sql_select.h: Auto merged sql/sql_test.cc: Auto merged sql/sql_update.cc: Auto merged sql/structs.h: Auto merged sql/table.cc: Auto merged sql/table.h: Auto merged sql/time.cc: Auto merged sql/unireg.cc: Auto merged sql/unireg.h: Auto merged storage/federated/ha_federated.cc: Auto merged storage/heap/Makefile.am: Auto merged storage/heap/_check.c: Auto merged storage/heap/_rectest.c: Auto merged storage/heap/ha_heap.cc: Auto merged storage/heap/ha_heap.h: Auto merged storage/heap/heapdef.h: Auto merged storage/heap/hp_block.c: Auto merged storage/heap/hp_clear.c: Auto merged storage/heap/hp_close.c: Auto merged storage/heap/hp_create.c: Auto merged storage/heap/hp_delete.c: Auto merged storage/heap/hp_extra.c: Auto merged storage/heap/hp_hash.c: Auto merged storage/heap/hp_info.c: Auto merged storage/heap/hp_open.c: Auto merged storage/heap/hp_panic.c: Auto merged storage/heap/hp_rename.c: Auto merged storage/heap/hp_rfirst.c: Auto merged storage/heap/hp_rkey.c: Auto merged storage/heap/hp_rlast.c: Auto merged storage/heap/hp_rnext.c: Auto merged storage/heap/hp_rprev.c: Auto merged storage/heap/hp_rrnd.c: Auto merged storage/heap/hp_rsame.c: Auto merged storage/heap/hp_scan.c: Auto merged storage/heap/hp_static.c: Auto merged storage/heap/hp_test1.c: Auto merged storage/heap/hp_test2.c: Auto merged storage/heap/hp_update.c: Auto merged storage/heap/hp_write.c: Auto merged storage/innobase/Makefile.am: Auto merged storage/innobase/btr/Makefile.am: Auto merged storage/innobase/buf/Makefile.am: Auto merged storage/innobase/data/Makefile.am: Auto merged storage/innobase/dict/Makefile.am: Auto merged storage/innobase/dyn/Makefile.am: Auto merged storage/innobase/eval/Makefile.am: Auto merged storage/innobase/fil/Makefile.am: Auto merged storage/innobase/fsp/Makefile.am: Auto merged storage/innobase/fut/Makefile.am: Auto merged storage/innobase/ha/Makefile.am: Auto merged storage/innobase/ibuf/Makefile.am: Auto merged storage/innobase/lock/Makefile.am: Auto merged storage/innobase/log/Makefile.am: Auto merged storage/innobase/mach/Makefile.am: Auto merged storage/innobase/mem/Makefile.am: Auto merged storage/innobase/mtr/Makefile.am: Auto merged storage/innobase/os/Makefile.am: Auto merged storage/innobase/page/Makefile.am: Auto merged storage/innobase/pars/Makefile.am: Auto merged storage/innobase/que/Makefile.am: Auto merged storage/innobase/read/Makefile.am: Auto merged storage/innobase/rem/Makefile.am: Auto merged storage/innobase/row/Makefile.am: Auto merged storage/innobase/srv/Makefile.am: Auto merged storage/innobase/sync/Makefile.am: Auto merged storage/innobase/thr/Makefile.am: Auto merged storage/innobase/trx/Makefile.am: Auto merged storage/innobase/usr/Makefile.am: Auto merged storage/innobase/ut/Makefile.am: Auto merged storage/myisam/Makefile.am: Auto merged storage/myisam/ft_boolean_search.c: Auto merged storage/myisam/ft_eval.c: Auto merged storage/myisam/ft_nlq_search.c: Auto merged storage/myisam/ft_parser.c: Auto merged storage/myisam/ft_static.c: Auto merged storage/myisam/ft_stem.c: Auto merged storage/myisam/ft_stopwords.c: Auto merged storage/myisam/ft_test1.c: Auto merged storage/myisam/ft_test1.h: Auto merged storage/myisam/ft_update.c: Auto merged storage/myisam/ftdefs.h: Auto merged storage/myisam/fulltext.h: Auto merged storage/myisam/ha_myisam.cc: Auto merged storage/myisam/ha_myisam.h: Auto merged storage/myisam/mi_cache.c: Auto merged storage/myisam/mi_changed.c: Auto merged storage/myisam/mi_check.c: Auto merged storage/myisam/mi_checksum.c: Auto merged storage/myisam/mi_close.c: Auto merged storage/myisam/mi_create.c: Auto merged storage/myisam/mi_dbug.c: Auto merged storage/myisam/mi_delete.c: Auto merged storage/myisam/mi_delete_all.c: Auto merged storage/myisam/mi_delete_table.c: Auto merged storage/myisam/mi_dynrec.c: Auto merged storage/myisam/mi_extra.c: Auto merged storage/myisam/mi_info.c: Auto merged storage/myisam/mi_key.c: Auto merged storage/myisam/mi_locking.c: Auto merged storage/myisam/mi_log.c: Auto merged storage/myisam/mi_open.c: Auto merged storage/myisam/mi_packrec.c: Auto merged storage/myisam/mi_page.c: Auto merged storage/myisam/mi_panic.c: Auto merged storage/myisam/mi_preload.c: Auto merged storage/myisam/mi_range.c: Auto merged storage/myisam/mi_rename.c: Auto merged storage/myisam/mi_rfirst.c: Auto merged storage/myisam/mi_rkey.c: Auto merged storage/myisam/mi_rlast.c: Auto merged storage/myisam/mi_rnext.c: Auto merged storage/myisam/mi_rnext_same.c: Auto merged storage/myisam/mi_rprev.c: Auto merged storage/myisam/mi_rrnd.c: Auto merged storage/myisam/mi_rsame.c: Auto merged storage/myisam/mi_rsamepos.c: Auto merged storage/myisam/mi_scan.c: Auto merged storage/myisam/mi_search.c: Auto merged storage/myisam/mi_static.c: Auto merged storage/myisam/mi_statrec.c: Auto merged storage/myisam/mi_test1.c: Auto merged storage/myisam/mi_test2.c: Auto merged storage/myisam/mi_test3.c: Auto merged storage/myisam/mi_unique.c: Auto merged storage/myisam/mi_update.c: Auto merged storage/myisam/mi_write.c: Auto merged storage/myisam/myisam_ftdump.c: Auto merged storage/myisam/myisamdef.h: Auto merged storage/myisam/myisamlog.c: Auto merged storage/myisam/myisampack.c: Auto merged storage/myisam/rt_index.c: Auto merged storage/myisam/rt_index.h: Auto merged storage/myisam/rt_key.h: Auto merged storage/myisam/rt_mbr.c: Auto merged storage/myisam/rt_mbr.h: Auto merged storage/myisam/rt_split.c: Auto merged storage/myisam/rt_test.c: Auto merged storage/myisam/sort.c: Auto merged storage/myisam/sp_defs.h: Auto merged storage/myisam/sp_test.c: Auto merged storage/myisammrg/Makefile.am: Auto merged storage/myisammrg/ha_myisammrg.cc: Auto merged storage/myisammrg/ha_myisammrg.h: Auto merged storage/myisammrg/myrg_close.c: Auto merged storage/myisammrg/myrg_create.c: Auto merged storage/myisammrg/myrg_def.h: Auto merged storage/myisammrg/myrg_delete.c: Auto merged storage/myisammrg/myrg_extra.c: Auto merged storage/myisammrg/myrg_info.c: Auto merged storage/myisammrg/myrg_locking.c: Auto merged storage/myisammrg/myrg_open.c: Auto merged storage/myisammrg/myrg_panic.c: Auto merged storage/myisammrg/myrg_queue.c: Auto merged storage/myisammrg/myrg_range.c: Auto merged storage/myisammrg/myrg_rfirst.c: Auto merged storage/myisammrg/myrg_rkey.c: Auto merged storage/myisammrg/myrg_rlast.c: Auto merged storage/myisammrg/myrg_rnext.c: Auto merged storage/myisammrg/myrg_rnext_same.c: Auto merged storage/myisammrg/myrg_rprev.c: Auto merged storage/myisammrg/myrg_rrnd.c: Auto merged storage/myisammrg/myrg_rsame.c: Auto merged storage/myisammrg/myrg_static.c: Auto merged storage/myisammrg/myrg_update.c: Auto merged storage/myisammrg/myrg_write.c: Auto merged storage/ndb/Makefile.am: Auto merged storage/ndb/config/common.mk.am: Auto merged storage/ndb/config/type_kernel.mk.am: Auto merged storage/ndb/config/type_mgmapiclient.mk.am: Auto merged storage/ndb/config/type_ndbapi.mk.am: Auto merged storage/ndb/config/type_ndbapiclient.mk.am: Auto merged storage/ndb/config/type_ndbapitest.mk.am: Auto merged storage/ndb/config/type_ndbapitools.mk.am: Auto merged storage/ndb/config/type_util.mk.am: Auto merged storage/ndb/docs/Makefile.am: Auto merged storage/ndb/include/Makefile.am: Auto merged storage/ndb/include/kernel/kernel_config_parameters.h: Auto merged storage/ndb/include/kernel/signaldata/CntrStart.hpp: Auto merged storage/ndb/include/kernel/signaldata/ReadConfig.hpp: Auto merged storage/ndb/include/kernel/signaldata/UpgradeStartup.hpp: Auto merged storage/ndb/include/logger/FileLogHandler.hpp: Auto merged storage/ndb/include/ndb_net.h: Auto merged storage/ndb/include/mgmapi/mgmapi_config_parameters.h: Auto merged storage/ndb/include/mgmapi/mgmapi_config_parameters_debug.h: Auto merged storage/ndb/include/util/ConfigValues.hpp: Auto merged storage/ndb/include/util/File.hpp: Auto merged storage/ndb/include/util/Vector.hpp: Auto merged storage/ndb/src/Makefile.am: Auto merged storage/ndb/src/common/Makefile.am: Auto merged storage/ndb/src/common/debugger/Makefile.am: Auto merged storage/ndb/src/common/debugger/signaldata/CntrStart.cpp: Auto merged storage/ndb/src/common/debugger/signaldata/Makefile.am: Auto merged storage/ndb/src/common/debugger/signaldata/ReadNodesConf.cpp: Auto merged storage/ndb/src/common/debugger/signaldata/print.awk: Auto merged storage/ndb/src/common/logger/FileLogHandler.cpp: Auto merged storage/ndb/src/common/logger/Makefile.am: Auto merged storage/ndb/src/common/mgmcommon/Makefile.am: Auto merged storage/ndb/src/common/transporter/Makefile.am: Auto merged storage/ndb/src/common/util/Bitmask.cpp: Auto merged storage/ndb/src/common/util/ConfigValues.cpp: Auto merged storage/ndb/src/common/util/File.cpp: Auto merged storage/ndb/src/common/util/Makefile.am: Auto merged storage/ndb/src/common/util/new.cpp: Auto merged storage/ndb/src/common/util/testConfigValues/testConfigValues.cpp: Auto merged storage/ndb/src/cw/Makefile.am: Auto merged storage/ndb/src/cw/cpcd/Makefile.am: Auto merged storage/ndb/src/kernel/blocks/Makefile.am: Auto merged storage/ndb/src/kernel/blocks/backup/Makefile.am: Auto merged storage/ndb/src/kernel/blocks/dbdict/Master_AddTable.sfl: Auto merged storage/ndb/src/kernel/blocks/dbdict/Slave_AddTable.sfl: Auto merged storage/ndb/src/kernel/blocks/dbtc/DbtcMain.cpp: Auto merged storage/ndb/src/kernel/error/Makefile.am: Auto merged storage/ndb/src/kernel/vm/Makefile.am: Auto merged storage/ndb/src/mgmapi/Makefile.am: Auto merged storage/ndb/src/mgmapi/mgmapi_configuration.cpp: Auto merged storage/ndb/src/mgmclient/Makefile.am: Auto merged storage/ndb/src/mgmsrv/Makefile.am: Auto merged storage/ndb/src/mgmsrv/MgmtSrvr.cpp: Auto merged storage/ndb/src/ndbapi/Makefile.am: Auto merged storage/ndb/src/ndbapi/NdbTransaction.cpp: Auto merged storage/ndb/src/ndbapi/Ndbif.cpp: Auto merged storage/ndb/test/Makefile.am: Auto merged storage/ndb/test/ndbapi/Makefile.am: Auto merged storage/ndb/test/ndbapi/bank/Makefile.am: Auto merged storage/ndb/test/ndbapi/testBasic.cpp: Auto merged storage/ndb/test/ndbapi/testIndex.cpp: Auto merged storage/ndb/test/run-test/Makefile.am: Auto merged storage/ndb/test/run-test/daily-basic-tests.txt: Auto merged storage/ndb/test/src/Makefile.am: Auto merged storage/ndb/test/tools/Makefile.am: Auto merged storage/ndb/tools/Makefile.am: Auto merged strings/Makefile.am: Auto merged strings/ctype-extra.c: Auto merged tests/Makefile.am: Auto merged vio/Makefile.am: Auto merged BitKeeper/deleted/.del-colspec-fix.pl~6c78d3332330b19e: Auto merged BitKeeper/deleted/.del-docbook-fixup.pl~46cf3bdef147084e: Auto merged BitKeeper/deleted/.del-docbook-prefix.pl~876c7d33c68c224a: Auto merged BitKeeper/deleted/.del-docbook-split~be931c3922898d0: Auto merged BitKeeper/deleted/.del-make-docbook~ccac1eb717e92ac9: Auto merged BitKeeper/deleted/.del-make-makefile~39fd454b487126e8: Auto merged BitKeeper/deleted/.del-test-make-manual-de~33cad2886311b8a: Auto merged BitKeeper/deleted/.del-test-make-manual~5da458f958a424ec: Auto merged BitKeeper/deleted/.del-xwf~76b97805d9146b80: Auto merged server-tools/instance-manager/listener.h: SCCS merged server-tools/instance-manager/manager.h: SCCS merged server-tools/instance-manager/mysql_connection.h: SCCS merged server-tools/instance-manager/priv.h: SCCS merged storage/ndb/src/kernel/blocks/dblqh/Makefile.am: SCCS merged
983 lines
27 KiB
C
983 lines
27 KiB
C
/* Copyright (C) 2002-2006 MySQL 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; version 2 of the License.
|
|
|
|
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 <my_global.h>
|
|
#include <m_string.h>
|
|
#include <stdlib.h>
|
|
#include <my_sys.h>
|
|
#include <mysys_err.h>
|
|
#include <my_getopt.h>
|
|
|
|
static void default_reporter(enum loglevel level, const char *format, ...);
|
|
my_error_reporter my_getopt_error_reporter= &default_reporter;
|
|
|
|
static int findopt(char *optpat, uint length,
|
|
const struct my_option **opt_res,
|
|
char **ffname);
|
|
my_bool getopt_compare_strings(const char *s,
|
|
const char *t,
|
|
uint length);
|
|
static longlong getopt_ll(char *arg, const struct my_option *optp, int *err);
|
|
static ulonglong getopt_ull(char *arg, const struct my_option *optp,
|
|
int *err);
|
|
static void init_variables(const struct my_option *options);
|
|
static int setval(const struct my_option *opts, gptr *value, char *argument,
|
|
my_bool set_maximum_value);
|
|
static char *check_struct_option(char *cur_arg, char *key_name);
|
|
|
|
/*
|
|
The following three variables belong to same group and the number and
|
|
order of their arguments must correspond to each other.
|
|
*/
|
|
static const char *special_opt_prefix[]=
|
|
{"skip", "disable", "enable", "maximum", "loose", 0};
|
|
static const uint special_opt_prefix_lengths[]=
|
|
{ 4, 7, 6, 7, 5, 0};
|
|
enum enum_special_opt
|
|
{ OPT_SKIP, OPT_DISABLE, OPT_ENABLE, OPT_MAXIMUM, OPT_LOOSE};
|
|
|
|
char *disabled_my_option= (char*) "0";
|
|
|
|
/*
|
|
This is a flag that can be set in client programs. 0 means that
|
|
my_getopt will not print error messages, but the client should do
|
|
it by itself
|
|
*/
|
|
|
|
my_bool my_getopt_print_errors= 1;
|
|
|
|
static void default_reporter(enum loglevel level,
|
|
const char *format, ...)
|
|
{
|
|
va_list args;
|
|
va_start(args, format);
|
|
if (level == WARNING_LEVEL)
|
|
fprintf(stderr, "%s", "Warning: ");
|
|
else if (level == INFORMATION_LEVEL)
|
|
fprintf(stderr, "%s", "Info: ");
|
|
vfprintf(stderr, format, args);
|
|
va_end(args);
|
|
fflush(stderr);
|
|
}
|
|
|
|
/*
|
|
function: handle_options
|
|
|
|
Sort options; put options first, until special end of options (--), or
|
|
until end of argv. Parse options; check that the given option matches with
|
|
one of the options in struct 'my_option', return error in case of ambiguous
|
|
or unknown option. Check that option was given an argument if it requires
|
|
one. Call function 'get_one_option()' once for each option.
|
|
*/
|
|
|
|
static gptr* (*getopt_get_addr)(const char *, uint, const struct my_option *);
|
|
|
|
void my_getopt_register_get_addr(gptr* (*func_addr)(const char *, uint,
|
|
const struct my_option *))
|
|
{
|
|
getopt_get_addr= func_addr;
|
|
}
|
|
|
|
int handle_options(int *argc, char ***argv,
|
|
const struct my_option *longopts,
|
|
my_get_one_option get_one_option)
|
|
{
|
|
uint opt_found, argvpos= 0, length, i;
|
|
my_bool end_of_options= 0, must_be_var, set_maximum_value,
|
|
option_is_loose;
|
|
char **pos, **pos_end, *optend, *prev_found,
|
|
*opt_str, key_name[FN_REFLEN];
|
|
const struct my_option *optp;
|
|
gptr *value;
|
|
int error;
|
|
|
|
LINT_INIT(opt_found);
|
|
(*argc)--; /* Skip the program name */
|
|
(*argv)++; /* --- || ---- */
|
|
init_variables(longopts);
|
|
|
|
for (pos= *argv, pos_end=pos+ *argc; pos != pos_end ; pos++)
|
|
{
|
|
char *cur_arg= *pos;
|
|
if (cur_arg[0] == '-' && cur_arg[1] && !end_of_options) /* must be opt */
|
|
{
|
|
char *argument= 0;
|
|
must_be_var= 0;
|
|
set_maximum_value= 0;
|
|
option_is_loose= 0;
|
|
|
|
cur_arg++; /* skip '-' */
|
|
if (*cur_arg == '-' || *cur_arg == 'O') /* check for long option, */
|
|
{ /* --set-variable, or -O */
|
|
if (*cur_arg == 'O')
|
|
{
|
|
must_be_var= 1;
|
|
|
|
if (!(*++cur_arg)) /* If not -Ovar=# */
|
|
{
|
|
/* the argument must be in next argv */
|
|
if (!*++pos)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: Option '-O' requires an argument\n",
|
|
my_progname);
|
|
return EXIT_ARGUMENT_REQUIRED;
|
|
}
|
|
cur_arg= *pos;
|
|
(*argc)--;
|
|
}
|
|
}
|
|
else if (!getopt_compare_strings(cur_arg, "-set-variable", 13))
|
|
{
|
|
must_be_var= 1;
|
|
if (cur_arg[13] == '=')
|
|
{
|
|
cur_arg+= 14;
|
|
if (!*cur_arg)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: Option '--set-variable' requires an argument\n",
|
|
my_progname);
|
|
return EXIT_ARGUMENT_REQUIRED;
|
|
}
|
|
}
|
|
else if (cur_arg[14]) /* garbage, or another option. break out */
|
|
must_be_var= 0;
|
|
else
|
|
{
|
|
/* the argument must be in next argv */
|
|
if (!*++pos)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: Option '--set-variable' requires an argument\n",
|
|
my_progname);
|
|
return EXIT_ARGUMENT_REQUIRED;
|
|
}
|
|
cur_arg= *pos;
|
|
(*argc)--;
|
|
}
|
|
}
|
|
else if (!must_be_var)
|
|
{
|
|
if (!*++cur_arg) /* skip the double dash */
|
|
{
|
|
/* '--' means end of options, look no further */
|
|
end_of_options= 1;
|
|
(*argc)--;
|
|
continue;
|
|
}
|
|
}
|
|
opt_str= check_struct_option(cur_arg, key_name);
|
|
optend= strcend(opt_str, '=');
|
|
length= (uint) (optend - opt_str);
|
|
if (*optend == '=')
|
|
optend++;
|
|
else
|
|
optend= 0;
|
|
|
|
/*
|
|
Find first the right option. Return error in case of an ambiguous,
|
|
or unknown option
|
|
*/
|
|
optp= longopts;
|
|
if (!(opt_found= findopt(opt_str, length, &optp, &prev_found)))
|
|
{
|
|
/*
|
|
Didn't find any matching option. Let's see if someone called
|
|
option with a special option prefix
|
|
*/
|
|
if (!must_be_var)
|
|
{
|
|
if (optend)
|
|
must_be_var= 1; /* option is followed by an argument */
|
|
for (i= 0; special_opt_prefix[i]; i++)
|
|
{
|
|
if (!getopt_compare_strings(special_opt_prefix[i], opt_str,
|
|
special_opt_prefix_lengths[i]) &&
|
|
(opt_str[special_opt_prefix_lengths[i]] == '-' ||
|
|
opt_str[special_opt_prefix_lengths[i]] == '_'))
|
|
{
|
|
/*
|
|
We were called with a special prefix, we can reuse opt_found
|
|
*/
|
|
opt_str+= (special_opt_prefix_lengths[i] + 1);
|
|
if (i == OPT_LOOSE)
|
|
option_is_loose= 1;
|
|
if ((opt_found= findopt(opt_str, length -
|
|
(special_opt_prefix_lengths[i] + 1),
|
|
&optp, &prev_found)))
|
|
{
|
|
if (opt_found > 1)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: ambiguous option '--%s-%s' (--%s-%s)\n",
|
|
my_progname, special_opt_prefix[i],
|
|
cur_arg, special_opt_prefix[i],
|
|
prev_found);
|
|
return EXIT_AMBIGUOUS_OPTION;
|
|
}
|
|
switch (i) {
|
|
case OPT_SKIP:
|
|
case OPT_DISABLE: /* fall through */
|
|
/*
|
|
double negation is actually enable again,
|
|
for example: --skip-option=0 -> option = TRUE
|
|
*/
|
|
optend= (optend && *optend == '0' && !(*(optend + 1))) ?
|
|
(char*) "1" : disabled_my_option;
|
|
break;
|
|
case OPT_ENABLE:
|
|
optend= (optend && *optend == '0' && !(*(optend + 1))) ?
|
|
disabled_my_option : (char*) "1";
|
|
break;
|
|
case OPT_MAXIMUM:
|
|
set_maximum_value= 1;
|
|
must_be_var= 1;
|
|
break;
|
|
}
|
|
break; /* break from the inner loop, main loop continues */
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (!opt_found)
|
|
{
|
|
if (must_be_var)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(option_is_loose ?
|
|
WARNING_LEVEL : ERROR_LEVEL,
|
|
"%s: unknown variable '%s'\n",
|
|
my_progname, cur_arg);
|
|
if (!option_is_loose)
|
|
return EXIT_UNKNOWN_VARIABLE;
|
|
}
|
|
else
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(option_is_loose ?
|
|
WARNING_LEVEL : ERROR_LEVEL,
|
|
"%s: unknown option '--%s'\n",
|
|
my_progname, cur_arg);
|
|
if (!option_is_loose)
|
|
return EXIT_UNKNOWN_OPTION;
|
|
}
|
|
if (option_is_loose)
|
|
{
|
|
(*argc)--;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
if (opt_found > 1)
|
|
{
|
|
if (must_be_var)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: variable prefix '%s' is not unique\n",
|
|
my_progname, opt_str);
|
|
return EXIT_VAR_PREFIX_NOT_UNIQUE;
|
|
}
|
|
else
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: ambiguous option '--%s' (%s, %s)\n",
|
|
my_progname, opt_str, prev_found,
|
|
optp->name);
|
|
return EXIT_AMBIGUOUS_OPTION;
|
|
}
|
|
}
|
|
if ((optp->var_type & GET_TYPE_MASK) == GET_DISABLED)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
fprintf(stderr,
|
|
"%s: %s: Option '%s' used, but is disabled\n", my_progname,
|
|
option_is_loose ? "WARNING" : "ERROR", opt_str);
|
|
if (option_is_loose)
|
|
{
|
|
(*argc)--;
|
|
continue;
|
|
}
|
|
return EXIT_OPTION_DISABLED;
|
|
}
|
|
if (must_be_var && (optp->var_type & GET_TYPE_MASK) == GET_NO_ARG)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: option '%s' cannot take an argument\n",
|
|
my_progname, optp->name);
|
|
return EXIT_NO_ARGUMENT_ALLOWED;
|
|
}
|
|
value= optp->var_type & GET_ASK_ADDR ?
|
|
(*getopt_get_addr)(key_name, (uint) strlen(key_name), optp) : optp->value;
|
|
|
|
if (optp->arg_type == NO_ARG)
|
|
{
|
|
if (optend && (optp->var_type & GET_TYPE_MASK) != GET_BOOL)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: option '--%s' cannot take an argument\n",
|
|
my_progname, optp->name);
|
|
return EXIT_NO_ARGUMENT_ALLOWED;
|
|
}
|
|
if ((optp->var_type & GET_TYPE_MASK) == GET_BOOL)
|
|
{
|
|
/*
|
|
Set bool to 1 if no argument or if the user has used
|
|
--enable-'option-name'.
|
|
*optend was set to '0' if one used --disable-option
|
|
*/
|
|
(*argc)--;
|
|
if (!optend || *optend == '1' ||
|
|
!my_strcasecmp(&my_charset_latin1, optend, "true"))
|
|
*((my_bool*) value)= (my_bool) 1;
|
|
else if (*optend == '0' ||
|
|
!my_strcasecmp(&my_charset_latin1, optend, "false"))
|
|
*((my_bool*) value)= (my_bool) 0;
|
|
else
|
|
{
|
|
my_getopt_error_reporter(WARNING_LEVEL,
|
|
"%s: ignoring option '--%s' due to \
|
|
invalid value '%s'\n",
|
|
my_progname, optp->name, optend);
|
|
continue;
|
|
}
|
|
get_one_option(optp->id, optp,
|
|
*((my_bool*) value) ?
|
|
(char*) "1" : disabled_my_option);
|
|
continue;
|
|
}
|
|
argument= optend;
|
|
}
|
|
else if (optp->arg_type == OPT_ARG &&
|
|
(optp->var_type & GET_TYPE_MASK) == GET_BOOL)
|
|
{
|
|
if (optend == disabled_my_option)
|
|
*((my_bool*) value)= (my_bool) 0;
|
|
else
|
|
{
|
|
if (!optend) /* No argument -> enable option */
|
|
*((my_bool*) value)= (my_bool) 1;
|
|
else
|
|
argument= optend;
|
|
}
|
|
}
|
|
else if (optp->arg_type == REQUIRED_ARG && !optend)
|
|
{
|
|
/* Check if there are more arguments after this one */
|
|
if (!*++pos)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: option '--%s' requires an argument\n",
|
|
my_progname, optp->name);
|
|
return EXIT_ARGUMENT_REQUIRED;
|
|
}
|
|
argument= *pos;
|
|
(*argc)--;
|
|
}
|
|
else
|
|
argument= optend;
|
|
}
|
|
else /* must be short option */
|
|
{
|
|
for (optend= cur_arg; *optend; optend++)
|
|
{
|
|
opt_found= 0;
|
|
for (optp= longopts; optp->id; optp++)
|
|
{
|
|
if (optp->id == (int) (uchar) *optend)
|
|
{
|
|
/* Option recognized. Find next what to do with it */
|
|
opt_found= 1;
|
|
if ((optp->var_type & GET_TYPE_MASK) == GET_DISABLED)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
fprintf(stderr,
|
|
"%s: ERROR: Option '-%c' used, but is disabled\n",
|
|
my_progname, optp->id);
|
|
return EXIT_OPTION_DISABLED;
|
|
}
|
|
if ((optp->var_type & GET_TYPE_MASK) == GET_BOOL &&
|
|
optp->arg_type == NO_ARG)
|
|
{
|
|
*((my_bool*) optp->value)= (my_bool) 1;
|
|
get_one_option(optp->id, optp, argument);
|
|
continue;
|
|
}
|
|
else if (optp->arg_type == REQUIRED_ARG ||
|
|
optp->arg_type == OPT_ARG)
|
|
{
|
|
if (*(optend + 1))
|
|
{
|
|
/* The rest of the option is option argument */
|
|
argument= optend + 1;
|
|
/* This is in effect a jump out of the outer loop */
|
|
optend= (char*) " ";
|
|
}
|
|
else
|
|
{
|
|
if (optp->arg_type == OPT_ARG)
|
|
{
|
|
if (optp->var_type == GET_BOOL)
|
|
*((my_bool*) optp->value)= (my_bool) 1;
|
|
get_one_option(optp->id, optp, argument);
|
|
continue;
|
|
}
|
|
/* Check if there are more arguments after this one */
|
|
if (!pos[1])
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: option '-%c' requires an argument\n",
|
|
my_progname, optp->id);
|
|
return EXIT_ARGUMENT_REQUIRED;
|
|
}
|
|
argument= *++pos;
|
|
(*argc)--;
|
|
/* the other loop will break, because *optend + 1 == 0 */
|
|
}
|
|
}
|
|
if ((error= setval(optp, optp->value, argument,
|
|
set_maximum_value)))
|
|
{
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: Error while setting value '%s' to '%s'\n",
|
|
my_progname, argument, optp->name);
|
|
return error;
|
|
}
|
|
get_one_option(optp->id, optp, argument);
|
|
break;
|
|
}
|
|
}
|
|
if (!opt_found)
|
|
{
|
|
if (my_getopt_print_errors)
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: unknown option '-%c'\n",
|
|
my_progname, *optend);
|
|
return EXIT_UNKNOWN_OPTION;
|
|
}
|
|
}
|
|
(*argc)--; /* option handled (short), decrease argument count */
|
|
continue;
|
|
}
|
|
if ((error= setval(optp, value, argument, set_maximum_value)))
|
|
{
|
|
my_getopt_error_reporter(ERROR_LEVEL,
|
|
"%s: Error while setting value '%s' to '%s'\n",
|
|
my_progname, argument, optp->name);
|
|
return error;
|
|
}
|
|
get_one_option(optp->id, optp, argument);
|
|
|
|
(*argc)--; /* option handled (short or long), decrease argument count */
|
|
}
|
|
else /* non-option found */
|
|
(*argv)[argvpos++]= cur_arg;
|
|
}
|
|
/*
|
|
Destroy the first, already handled option, so that programs that look
|
|
for arguments in 'argv', without checking 'argc', know when to stop.
|
|
Items in argv, before the destroyed one, are all non-option -arguments
|
|
to the program, yet to be (possibly) handled.
|
|
*/
|
|
(*argv)[argvpos]= 0;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
function: check_struct_option
|
|
|
|
Arguments: Current argument under processing from argv and a variable
|
|
where to store the possible key name.
|
|
|
|
Return value: In case option is a struct option, returns a pointer to
|
|
the current argument at the position where the struct option (key_name)
|
|
ends, the next character after the dot. In case argument is not a struct
|
|
option, returns a pointer to the argument.
|
|
|
|
key_name will hold the name of the key, or 0 if not found.
|
|
*/
|
|
|
|
static char *check_struct_option(char *cur_arg, char *key_name)
|
|
{
|
|
char *ptr, *end;
|
|
|
|
ptr= strcend(cur_arg + 1, '.'); /* Skip the first character */
|
|
end= strcend(cur_arg, '=');
|
|
|
|
/*
|
|
If the first dot is after an equal sign, then it is part
|
|
of a variable value and the option is not a struct option.
|
|
Also, if the last character in the string before the ending
|
|
NULL, or the character right before equal sign is the first
|
|
dot found, the option is not a struct option.
|
|
*/
|
|
if (end - ptr > 1)
|
|
{
|
|
uint len= (uint) (ptr - cur_arg);
|
|
set_if_smaller(len, FN_REFLEN-1);
|
|
strmake(key_name, cur_arg, len);
|
|
return ++ptr;
|
|
}
|
|
else
|
|
{
|
|
key_name[0]= 0;
|
|
return cur_arg;
|
|
}
|
|
}
|
|
|
|
/*
|
|
function: setval
|
|
|
|
Arguments: opts, argument
|
|
Will set the option value to given value
|
|
*/
|
|
|
|
static int setval(const struct my_option *opts, gptr *value, char *argument,
|
|
my_bool set_maximum_value)
|
|
{
|
|
int err= 0;
|
|
|
|
if (value && argument)
|
|
{
|
|
gptr *result_pos= ((set_maximum_value) ?
|
|
opts->u_max_value : value);
|
|
|
|
if (!result_pos)
|
|
return EXIT_NO_PTR_TO_VARIABLE;
|
|
|
|
switch ((opts->var_type & GET_TYPE_MASK)) {
|
|
case GET_BOOL: /* If argument differs from 0, enable option, else disable */
|
|
*((my_bool*) result_pos)= (my_bool) atoi(argument) != 0;
|
|
break;
|
|
case GET_INT:
|
|
case GET_UINT: /* fall through */
|
|
*((int*) result_pos)= (int) getopt_ll(argument, opts, &err);
|
|
break;
|
|
case GET_LONG:
|
|
case GET_ULONG: /* fall through */
|
|
*((long*) result_pos)= (long) getopt_ll(argument, opts, &err);
|
|
break;
|
|
case GET_LL:
|
|
*((longlong*) result_pos)= getopt_ll(argument, opts, &err);
|
|
break;
|
|
case GET_ULL:
|
|
*((ulonglong*) result_pos)= getopt_ull(argument, opts, &err);
|
|
break;
|
|
case GET_STR:
|
|
*((char**) result_pos)= argument;
|
|
break;
|
|
case GET_STR_ALLOC:
|
|
if ((*((char**) result_pos)))
|
|
my_free((*(char**) result_pos), MYF(MY_WME | MY_FAE));
|
|
if (!(*((char**) result_pos)= my_strdup(argument, MYF(MY_WME))))
|
|
return EXIT_OUT_OF_MEMORY;
|
|
break;
|
|
default: /* dummy default to avoid compiler warnings */
|
|
break;
|
|
}
|
|
if (err)
|
|
return EXIT_UNKNOWN_SUFFIX;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
Find option
|
|
|
|
SYNOPSIS
|
|
findopt()
|
|
optpat Prefix of option to find (with - or _)
|
|
length Length of optpat
|
|
opt_res Options
|
|
ffname Place for pointer to first found name
|
|
|
|
IMPLEMENTATION
|
|
Go through all options in the my_option struct. Return number
|
|
of options found that match the pattern and in the argument
|
|
list the option found, if any. In case of ambiguous option, store
|
|
the name in ffname argument
|
|
|
|
RETURN
|
|
0 No matching options
|
|
# Number of matching options
|
|
ffname points to first matching option
|
|
*/
|
|
|
|
static int findopt(char *optpat, uint length,
|
|
const struct my_option **opt_res,
|
|
char **ffname)
|
|
{
|
|
uint count;
|
|
struct my_option *opt= (struct my_option *) *opt_res;
|
|
|
|
for (count= 0; opt->name; opt++)
|
|
{
|
|
if (!getopt_compare_strings(opt->name, optpat, length)) /* match found */
|
|
{
|
|
(*opt_res)= opt;
|
|
if (!opt->name[length]) /* Exact match */
|
|
return 1;
|
|
if (!count)
|
|
{
|
|
count= 1;
|
|
*ffname= (char *) opt->name; /* We only need to know one prev */
|
|
}
|
|
else if (strcmp(*ffname, opt->name))
|
|
{
|
|
/*
|
|
The above test is to not count same option twice
|
|
(see mysql.cc, option "help")
|
|
*/
|
|
count++;
|
|
}
|
|
}
|
|
}
|
|
return count;
|
|
}
|
|
|
|
|
|
/*
|
|
function: compare_strings
|
|
|
|
Works like strncmp, other than 1.) considers '-' and '_' the same.
|
|
2.) Returns -1 if strings differ, 0 if they are equal
|
|
*/
|
|
|
|
my_bool getopt_compare_strings(register const char *s, register const char *t,
|
|
uint length)
|
|
{
|
|
char const *end= s + length;
|
|
for (;s != end ; s++, t++)
|
|
{
|
|
if ((*s != '-' ? *s : '_') != (*t != '-' ? *t : '_'))
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
function: eval_num_suffix
|
|
|
|
Transforms a number with a suffix to real number. Suffix can
|
|
be k|K for kilo, m|M for mega or g|G for giga.
|
|
*/
|
|
|
|
static longlong eval_num_suffix (char *argument, int *error, char *option_name)
|
|
{
|
|
char *endchar;
|
|
longlong num;
|
|
|
|
*error= 0;
|
|
num= strtoll(argument, &endchar, 10);
|
|
if (*endchar == 'k' || *endchar == 'K')
|
|
num*= 1024L;
|
|
else if (*endchar == 'm' || *endchar == 'M')
|
|
num*= 1024L * 1024L;
|
|
else if (*endchar == 'g' || *endchar == 'G')
|
|
num*= 1024L * 1024L * 1024L;
|
|
else if (*endchar)
|
|
{
|
|
fprintf(stderr,
|
|
"Unknown suffix '%c' used for variable '%s' (value '%s')\n",
|
|
*endchar, option_name, argument);
|
|
*error= 1;
|
|
return 0;
|
|
}
|
|
return num;
|
|
}
|
|
|
|
/*
|
|
function: getopt_ll
|
|
|
|
Evaluates and returns the value that user gave as an argument
|
|
to a variable. Recognizes (case insensitive) K as KILO, M as MEGA
|
|
and G as GIGA bytes. Some values must be in certain blocks, as
|
|
defined in the given my_option struct, this function will check
|
|
that those values are honored.
|
|
In case of an error, set error value in *err.
|
|
*/
|
|
|
|
static longlong getopt_ll(char *arg, const struct my_option *optp, int *err)
|
|
{
|
|
longlong num;
|
|
ulonglong block_size= (optp->block_size ? (ulonglong) optp->block_size : 1L);
|
|
|
|
num= eval_num_suffix(arg, err, (char*) optp->name);
|
|
if (num > 0 && (ulonglong) num > (ulonglong) optp->max_value &&
|
|
optp->max_value) /* if max value is not set -> no upper limit */
|
|
num= (ulonglong) optp->max_value;
|
|
num= ((num - optp->sub_size) / block_size);
|
|
num= (longlong) (num * block_size);
|
|
return max(num, optp->min_value);
|
|
}
|
|
|
|
/*
|
|
function: getopt_ull
|
|
|
|
This is the same as getopt_ll, but is meant for unsigned long long
|
|
values.
|
|
*/
|
|
|
|
static ulonglong getopt_ull(char *arg, const struct my_option *optp, int *err)
|
|
{
|
|
ulonglong num;
|
|
|
|
num= eval_num_suffix(arg, err, (char*) optp->name);
|
|
return getopt_ull_limit_value(num, optp);
|
|
}
|
|
|
|
|
|
ulonglong getopt_ull_limit_value(ulonglong num, const struct my_option *optp)
|
|
{
|
|
if ((ulonglong) num > (ulonglong) optp->max_value &&
|
|
optp->max_value) /* if max value is not set -> no upper limit */
|
|
num= (ulonglong) optp->max_value;
|
|
if (optp->block_size > 1)
|
|
{
|
|
num/= (ulonglong) optp->block_size;
|
|
num*= (ulonglong) optp->block_size;
|
|
}
|
|
if (num < (ulonglong) optp->min_value)
|
|
num= (ulonglong) optp->min_value;
|
|
return num;
|
|
}
|
|
|
|
|
|
/*
|
|
Init one value to it's default values
|
|
|
|
SYNOPSIS
|
|
init_one_value()
|
|
option Option to initialize
|
|
value Pointer to variable
|
|
*/
|
|
|
|
static void init_one_value(const struct my_option *option, gptr *variable,
|
|
longlong value)
|
|
{
|
|
switch ((option->var_type & GET_TYPE_MASK)) {
|
|
case GET_BOOL:
|
|
*((my_bool*) variable)= (my_bool) value;
|
|
break;
|
|
case GET_INT:
|
|
*((int*) variable)= (int) value;
|
|
break;
|
|
case GET_UINT:
|
|
*((uint*) variable)= (uint) value;
|
|
break;
|
|
case GET_LONG:
|
|
*((long*) variable)= (long) value;
|
|
break;
|
|
case GET_ULONG:
|
|
*((ulong*) variable)= (ulong) value;
|
|
break;
|
|
case GET_LL:
|
|
*((longlong*) variable)= (longlong) value;
|
|
break;
|
|
case GET_ULL:
|
|
*((ulonglong*) variable)= (ulonglong) value;
|
|
break;
|
|
default: /* dummy default to avoid compiler warnings */
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
initialize all variables to their default values
|
|
|
|
SYNOPSIS
|
|
init_variables()
|
|
options Array of options
|
|
|
|
NOTES
|
|
We will initialize the value that is pointed to by options->value.
|
|
If the value is of type GET_ASK_ADDR, we will also ask for the address
|
|
for a value and initialize.
|
|
*/
|
|
|
|
static void init_variables(const struct my_option *options)
|
|
{
|
|
for (; options->name; options++)
|
|
{
|
|
gptr *variable;
|
|
/*
|
|
We must set u_max_value first as for some variables
|
|
options->u_max_value == options->value and in this case we want to
|
|
set the value to default value.
|
|
*/
|
|
if (options->u_max_value)
|
|
init_one_value(options, options->u_max_value, options->max_value);
|
|
if (options->value)
|
|
init_one_value(options, options->value, options->def_value);
|
|
if (options->var_type & GET_ASK_ADDR &&
|
|
(variable= (*getopt_get_addr)("", 0, options)))
|
|
init_one_value(options, variable, options->def_value);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
function: my_print_options
|
|
|
|
Print help for all options and variables.
|
|
*/
|
|
|
|
#include <help_start.h>
|
|
|
|
void my_print_help(const struct my_option *options)
|
|
{
|
|
uint col, name_space= 22, comment_space= 57;
|
|
const char *line_end;
|
|
const struct my_option *optp;
|
|
|
|
for (optp= options; optp->id; optp++)
|
|
{
|
|
if (optp->id < 256)
|
|
{
|
|
printf(" -%c%s", optp->id, strlen(optp->name) ? ", " : " ");
|
|
col= 6;
|
|
}
|
|
else
|
|
{
|
|
printf(" ");
|
|
col= 2;
|
|
}
|
|
if (strlen(optp->name))
|
|
{
|
|
printf("--%s", optp->name);
|
|
col+= 2 + (uint) strlen(optp->name);
|
|
if ((optp->var_type & GET_TYPE_MASK) == GET_STR ||
|
|
(optp->var_type & GET_TYPE_MASK) == GET_STR_ALLOC)
|
|
{
|
|
printf("%s=name%s ", optp->arg_type == OPT_ARG ? "[" : "",
|
|
optp->arg_type == OPT_ARG ? "]" : "");
|
|
col+= (optp->arg_type == OPT_ARG) ? 8 : 6;
|
|
}
|
|
else if ((optp->var_type & GET_TYPE_MASK) == GET_NO_ARG ||
|
|
(optp->var_type & GET_TYPE_MASK) == GET_BOOL)
|
|
{
|
|
putchar(' ');
|
|
col++;
|
|
}
|
|
else
|
|
{
|
|
printf("%s=#%s ", optp->arg_type == OPT_ARG ? "[" : "",
|
|
optp->arg_type == OPT_ARG ? "]" : "");
|
|
col+= (optp->arg_type == OPT_ARG) ? 5 : 3;
|
|
}
|
|
if (col > name_space && optp->comment && *optp->comment)
|
|
{
|
|
putchar('\n');
|
|
col= 0;
|
|
}
|
|
}
|
|
for (; col < name_space; col++)
|
|
putchar(' ');
|
|
if (optp->comment && *optp->comment)
|
|
{
|
|
const char *comment= optp->comment, *end= strend(comment);
|
|
|
|
while ((uint) (end - comment) > comment_space)
|
|
{
|
|
for (line_end= comment + comment_space; *line_end != ' '; line_end--);
|
|
for (; comment != line_end; comment++)
|
|
putchar(*comment);
|
|
comment++; /* skip the space, as a newline will take it's place now */
|
|
putchar('\n');
|
|
for (col= 0; col < name_space; col++)
|
|
putchar(' ');
|
|
}
|
|
printf("%s", comment);
|
|
}
|
|
putchar('\n');
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
function: my_print_options
|
|
|
|
Print variables.
|
|
*/
|
|
|
|
void my_print_variables(const struct my_option *options)
|
|
{
|
|
uint name_space= 34, length;
|
|
char buff[255];
|
|
const struct my_option *optp;
|
|
|
|
printf("\nVariables (--variable-name=value)\n");
|
|
printf("and boolean options {FALSE|TRUE} Value (after reading options)\n");
|
|
printf("--------------------------------- -----------------------------\n");
|
|
for (optp= options; optp->id; optp++)
|
|
{
|
|
gptr *value= (optp->var_type & GET_ASK_ADDR ?
|
|
(*getopt_get_addr)("", 0, optp) : optp->value);
|
|
if (value)
|
|
{
|
|
printf("%s ", optp->name);
|
|
length= (uint) strlen(optp->name)+1;
|
|
for (; length < name_space; length++)
|
|
putchar(' ');
|
|
switch ((optp->var_type & GET_TYPE_MASK)) {
|
|
case GET_STR:
|
|
case GET_STR_ALLOC: /* fall through */
|
|
printf("%s\n", *((char**) value) ? *((char**) value) :
|
|
"(No default value)");
|
|
break;
|
|
case GET_BOOL:
|
|
printf("%s\n", *((my_bool*) value) ? "TRUE" : "FALSE");
|
|
break;
|
|
case GET_INT:
|
|
printf("%d\n", *((int*) value));
|
|
break;
|
|
case GET_UINT:
|
|
printf("%d\n", *((uint*) value));
|
|
break;
|
|
case GET_LONG:
|
|
printf("%lu\n", *((long*) value));
|
|
break;
|
|
case GET_ULONG:
|
|
printf("%lu\n", *((ulong*) value));
|
|
break;
|
|
case GET_LL:
|
|
printf("%s\n", llstr(*((longlong*) value), buff));
|
|
break;
|
|
case GET_ULL:
|
|
longlong2str(*((ulonglong*) value), buff, 10);
|
|
printf("%s\n", buff);
|
|
break;
|
|
default:
|
|
printf("(Disabled)\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#include <help_end.h>
|