Reduce overhead of setting fish vars

The `react_to_variable_change()` function is called whenever a fish var
is set. Even as a consequence of statements like `for x in a b c`. It is
therefore critical that that function be as fast as possible. Especially
when setting the var doesn't have any side-effects which is true something
like 99.9999% of the time.

This change reduces the overhead of `react_to_variable_change()` to
unmeasurable levels. Making the synthetic benchmark in issue #4341
36% faster.

Fixes #4341
This commit is contained in:
Kurtis Rader 2017-08-18 17:26:45 -07:00
parent 55b2d36028
commit b1ac07a178
2 changed files with 92 additions and 46 deletions

View file

@ -16,6 +16,7 @@ This section is for changes merged to the `major` branch that are not also merge
- `complete` now has a `-k` and `--keep-order` option to keep the order of the OPTION_ARGUMENTS (#361). - `complete` now has a `-k` and `--keep-order` option to keep the order of the OPTION_ARGUMENTS (#361).
- Local exported (`set -lx`) vars are now visible to functions (#1091). - Local exported (`set -lx`) vars are now visible to functions (#1091).
- `abbr` has been reimplemented to be faster. This means the old `fish_user_abbreviations` variable is ignored (#4048). - `abbr` has been reimplemented to be faster. This means the old `fish_user_abbreviations` variable is ignored (#4048).
- Setting variables is much faster meaning fish is much faster (#4200, #4341).
## Other significant changes ## Other significant changes
- Command substitution output is now limited to 10 MB by default (#3822). - Command substitution output is now limited to 10 MB by default (#3822).

View file

@ -85,6 +85,9 @@ bool term_has_xn = false;
/// found in `TERMINFO_DIRS` we don't to call `handle_curses()` before we've imported the latter. /// found in `TERMINFO_DIRS` we don't to call `handle_curses()` before we've imported the latter.
static bool env_initialized = false; static bool env_initialized = false;
typedef std::map<const wcstring, void (*)(const wcstring &, const wcstring &)> var_dispatch_table_t;
var_dispatch_table_t var_dispatch_table;
/// List of all locale environment variable names that might trigger (re)initializing the locale /// List of all locale environment variable names that might trigger (re)initializing the locale
/// subsystem. /// subsystem.
static const wcstring_list_t locale_variables({L"LANG", L"LANGUAGE", L"LC_ALL", L"LC_ADDRESS", static const wcstring_list_t locale_variables({L"LANG", L"LANGUAGE", L"LC_ALL", L"LC_ADDRESS",
@ -348,9 +351,6 @@ static mode_t get_umask() {
return res; return res;
} }
/// Check if the specified variable is a timezone variable.
static bool var_is_timezone(const wcstring &key) { return key == L"TZ"; }
/// Properly sets all timezone information. /// Properly sets all timezone information.
static void handle_timezone(const wchar_t *env_var_name) { static void handle_timezone(const wchar_t *env_var_name) {
// const env_var_t var = env_get(env_var_name, ENV_EXPORT); // const env_var_t var = env_get(env_var_name, ENV_EXPORT);
@ -399,16 +399,6 @@ static void fix_colon_delimited_var(const wcstring &var_name) {
} }
} }
/// Check if the specified variable is a locale variable.
static bool var_is_locale(const wcstring &key) {
for (auto var_name : locale_variables) {
if (key == var_name) {
return true;
}
}
return false;
}
/// Initialize the locale subsystem. /// Initialize the locale subsystem.
static void init_locale() { static void init_locale() {
// We have to make a copy because the subsequent setlocale() call to change the locale will // We have to make a copy because the subsequent setlocale() call to change the locale will
@ -445,16 +435,6 @@ static void init_locale() {
free(old_msg_locale); free(old_msg_locale);
} }
/// Check if the specified variable is a locale variable.
static bool var_is_curses(const wcstring &key) {
for (auto var_name : curses_variables) {
if (key == var_name) {
return true;
}
}
return false;
}
/// True if we think we can set the terminal title else false. /// True if we think we can set the terminal title else false.
static bool can_set_term_title = false; static bool can_set_term_title = false;
@ -629,33 +609,18 @@ static bool variable_is_colon_delimited_var(const wcstring &str) {
static void react_to_variable_change(const wchar_t *op, const wcstring &key) { static void react_to_variable_change(const wchar_t *op, const wcstring &key) {
// Don't do any of this until `env_init()` has run. We only want to do this in response to // Don't do any of this until `env_init()` has run. We only want to do this in response to
// variables set by the user; e.g., in a script like *config.fish* or interactively or as part // variables set by the user; e.g., in a script like *config.fish* or interactively or as part
// of loading the universal variables for the first time. // of loading the universal variables for the first time. Variables we import from the
// environment or that are otherwise set by fish before this gets called have to explicitly
// call the appropriate functions to put the value of the var into effect.
if (!env_initialized) return; if (!env_initialized) return;
if (var_is_locale(key)) { auto dispatch = var_dispatch_table.find(key);
init_locale(); if (dispatch != var_dispatch_table.end()) {
} else if (variable_is_colon_delimited_var(key)) { (*dispatch->second)(op, key);
fix_colon_delimited_var(key); } else if (string_prefixes_string(L"_fish_abbr_", key)) {
} else if (var_is_curses(key)) { update_abbr_cache(op, key);
init_curses();
init_input();
} else if (var_is_timezone(key)) {
handle_timezone(key.c_str());
} else if (key == L"fish_term256" || key == L"fish_term24bit") {
update_fish_color_support();
reader_react_to_color_change();
} else if (string_prefixes_string(L"fish_color_", key)) { } else if (string_prefixes_string(L"fish_color_", key)) {
reader_react_to_color_change(); reader_react_to_color_change();
} else if (key == L"fish_escape_delay_ms") {
update_wait_on_escape_ms();
} else if (key == L"LINES" || key == L"COLUMNS") {
invalidate_termsize(true); // force fish to update its idea of the terminal size plus vars
} else if (key == L"FISH_READ_BYTE_LIMIT") {
env_set_read_limit();
} else if (key == L"FISH_HISTORY") {
reader_change_history(history_session_id().c_str());
} else if (wcsncmp(key.c_str(), L"_fish_abbr_", wcslen(L"_fish_abbr_")) == 0) {
update_abbr_cache(op, key);
} }
} }
@ -813,7 +778,87 @@ void env_universal_barrier() {
env_universal_callbacks(callbacks); env_universal_callbacks(callbacks);
} }
static void handle_fish_term_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
UNUSED(var_name);
update_fish_color_support();
reader_react_to_color_change();
}
static void handle_escape_delay_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
UNUSED(var_name);
update_wait_on_escape_ms();
}
static void handle_term_size_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
UNUSED(var_name);
invalidate_termsize(true); // force fish to update its idea of the terminal size plus vars
}
static void handle_read_limit_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
UNUSED(var_name);
env_set_read_limit();
}
static void handle_fish_history_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
UNUSED(var_name);
reader_change_history(history_session_id().c_str());
}
static void handle_tz_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
handle_timezone(var_name.c_str());
}
static void handle_magic_colon_var_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
fix_colon_delimited_var(var_name);
}
static void handle_locale_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
UNUSED(var_name);
init_locale();
}
static void handle_curses_change(const wcstring &op, const wcstring &var_name) {
UNUSED(op);
UNUSED(var_name);
init_curses();
}
/// Populate the dispatch table used by `react_to_variable_change()` to efficiently call the
/// appropriate function to handle a change to a variable.
static void setup_var_dispatch_table() {
for (auto var_name : locale_variables) {
var_dispatch_table.emplace(var_name, handle_locale_change);
}
for (auto var_name : curses_variables) {
var_dispatch_table.emplace(var_name, handle_curses_change);
}
for (auto var_name : colon_delimited_variable) {
var_dispatch_table.emplace(var_name, handle_magic_colon_var_change);
}
var_dispatch_table.emplace(L"fish_term256", handle_fish_term_change);
var_dispatch_table.emplace(L"fish_term24bit", handle_fish_term_change);
var_dispatch_table.emplace(L"fish_escape_delay_ms", handle_escape_delay_change);
var_dispatch_table.emplace(L"LINES", handle_term_size_change);
var_dispatch_table.emplace(L"COLUMNS", handle_term_size_change);
var_dispatch_table.emplace(L"FISH_READ_BYTE_LIMIT", handle_read_limit_change);
var_dispatch_table.emplace(L"FISH_HISTORY", handle_fish_history_change);
var_dispatch_table.emplace(L"TZ", handle_tz_change);
}
void env_init(const struct config_paths_t *paths /* or NULL */) { void env_init(const struct config_paths_t *paths /* or NULL */) {
setup_var_dispatch_table();
// These variables can not be altered directly by the user. // These variables can not be altered directly by the user.
const wchar_t *const ro_keys[] = { const wchar_t *const ro_keys[] = {
L"status", L"history", L"_", L"PWD", L"FISH_VERSION", L"status", L"history", L"_", L"PWD", L"FISH_VERSION",