Stop using a static unordered_map for string flag handlers

This switches the flag_to_function from a map to just an ordinary switch
statement. This saves some memory/startup time and removes some
relocations. No functional change here.
This commit is contained in:
ridiculousfish 2022-07-04 13:40:55 -07:00
parent aec8413f7b
commit 61b09ff4a7

View file

@ -269,6 +269,9 @@ static int handle_flag_1(const wchar_t **argv, parser_t &parser, io_streams_t &s
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;
} }
using flag_handler_t = int (*)(const wchar_t **argv, parser_t &parser, io_streams_t &streams,
const wgetopter_t &w, options_t *opts);
static int handle_flag_N(const wchar_t **argv, parser_t &parser, io_streams_t &streams, static int handle_flag_N(const wchar_t **argv, parser_t &parser, io_streams_t &streams,
const wgetopter_t &w, options_t *opts) { const wgetopter_t &w, options_t *opts) {
if (opts->no_newline_valid) { if (opts->no_newline_valid) {
@ -537,7 +540,7 @@ static int handle_flag_V(const wchar_t **argv, parser_t &parser, io_streams_t &s
static int handle_flag_w(const wchar_t **argv, parser_t &parser, io_streams_t &streams, static int handle_flag_w(const wchar_t **argv, parser_t &parser, io_streams_t &streams,
const wgetopter_t &w, options_t *opts) { const wgetopter_t &w, options_t *opts) {
if (opts->width_valid) { if (opts->width_valid) {
long width = fish_wcstol(w.woptarg); long width = fish_wcstol(w.woptarg);
if (width < 0) { if (width < 0) {
string_error(streams, _(L"%ls: Invalid width value '%ls'\n"), argv[0], w.woptarg); string_error(streams, _(L"%ls: Invalid width value '%ls'\n"), argv[0], w.woptarg);
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;
@ -617,12 +620,30 @@ static const struct woption long_options[] = {{L"all", no_argument, nullptr, 'a'
{L"width", required_argument, nullptr, 'w'}, {L"width", required_argument, nullptr, 'w'},
{}}; {}};
static const std::unordered_map<char, decltype(*handle_flag_N)> flag_to_function = { static flag_handler_t get_handler_for_flag(char c) {
{'N', handle_flag_N}, {'a', handle_flag_a}, {'c', handle_flag_c}, {'e', handle_flag_e}, // clang-format off
{'f', handle_flag_f}, {'g', handle_flag_g}, {'i', handle_flag_i}, {'l', handle_flag_l}, switch (c) {
{'m', handle_flag_m}, {'n', handle_flag_n}, {'q', handle_flag_q}, {'r', handle_flag_r}, case 'N': return handle_flag_N;
{'s', handle_flag_s}, {'V', handle_flag_V}, {'v', handle_flag_v}, {'w', handle_flag_w}, case 'a': return handle_flag_a;
{1, handle_flag_1}}; case 'c': return handle_flag_c;
case 'e': return handle_flag_e;
case 'f': return handle_flag_f;
case 'g': return handle_flag_g;
case 'i': return handle_flag_i;
case 'l': return handle_flag_l;
case 'm': return handle_flag_m;
case 'n': return handle_flag_n;
case 'q': return handle_flag_q;
case 'r': return handle_flag_r;
case 's': return handle_flag_s;
case 'V': return handle_flag_V;
case 'v': return handle_flag_v;
case 'w': return handle_flag_w;
case 1 : return handle_flag_1;
default: return nullptr;
}
// clang-format on
}
/// Parse the arguments for flags recognized by a specific string subcommand. /// Parse the arguments for flags recognized by a specific string subcommand.
static int parse_opts(options_t *opts, int *optind, int n_req_args, int argc, const wchar_t **argv, static int parse_opts(options_t *opts, int *optind, int n_req_args, int argc, const wchar_t **argv,
@ -633,9 +654,8 @@ static int parse_opts(options_t *opts, int *optind, int n_req_args, int argc, co
int opt; int opt;
wgetopter_t w; wgetopter_t w;
while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, nullptr)) != -1) { while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, nullptr)) != -1) {
auto fn = flag_to_function.find(opt); if (auto fn = get_handler_for_flag(opt)) {
if (fn != flag_to_function.end()) { int retval = fn(argv, parser, streams, w, opts);
int retval = fn->second(argv, parser, streams, w, opts);
if (retval != STATUS_CMD_OK) return retval; if (retval != STATUS_CMD_OK) return retval;
} else if (opt == ':') { } else if (opt == ':') {
streams.err.append(L"string "); // clone of string_error streams.err.append(L"string "); // clone of string_error