diff options
author | Phil Jones <philj56@gmail.com> | 2022-11-28 22:19:16 +0000 |
---|---|---|
committer | Phil Jones <philj56@gmail.com> | 2022-11-28 22:19:16 +0000 |
commit | dd36bf1c53216e1828d136cb735d65816575571f (patch) | |
tree | e722af346c019942b296e5aae6e8846afb15ed67 /src/compgen.c | |
parent | 3861e8289ae40bac168275ce6f10b231e11baa55 (diff) |
Improve performance of tofi-run history sorting.
Diffstat (limited to 'src/compgen.c')
-rw-r--r-- | src/compgen.c | 27 |
1 files changed, 25 insertions, 2 deletions
diff --git a/src/compgen.c b/src/compgen.c index 2fb2a04..e8422b7 100644 --- a/src/compgen.c +++ b/src/compgen.c @@ -234,7 +234,7 @@ static int cmpscorep(const void *restrict a, const void *restrict b) return str2->history_score - str1->history_score; } -void compgen_history_sort(struct string_ref_vec *programs, struct history *history) +struct string_ref_vec compgen_history_sort(struct string_ref_vec *programs, struct history *history) { log_debug("Moving already known programs to the front.\n"); for (size_t i = 0; i < history->count; i++) { @@ -246,5 +246,28 @@ void compgen_history_sort(struct string_ref_vec *programs, struct history *histo res->history_score = history->buf[i].run_count; } - qsort(programs->buf, programs->count, sizeof(programs->buf[0]), cmpscorep); + /* + * For compgen, we expect there to be many more commands than history + * entries. For speed, we therefore create a copy of the command + * vector with all of the non-zero history score items pushed to the + * front. We can then call qsort() on just the first few items of the + * new vector, rather than on the entire original vector. + */ + struct string_ref_vec vec = { + .count = programs->count, + .size = programs->size, + .buf = xcalloc(programs->size, sizeof(*vec.buf)) + }; + + size_t n_hist = 0; + for (ssize_t i = programs->count - 1; i >= 0; i--) { + if (programs->buf[i].history_score == 0) { + vec.buf[i + n_hist] = programs->buf[i]; + } else { + vec.buf[n_hist] = programs->buf[i]; + n_hist++; + } + } + qsort(vec.buf, n_hist, sizeof(vec.buf[0]), cmpscorep); + return vec; } |