From: Daniel Mendler Date: Mon, 19 Apr 2021 06:43:41 +0000 (+0200) Subject: completion-all-sorted-completions: Fix sorting performance bug X-Git-Tag: emacs-28.0.90~2813 X-Git-Url: http://git.eshelyaron.com/gitweb/?a=commitdiff_plain;h=18d0ef9d597c1a78a0063b5c04642bf4c448dd19;p=emacs.git completion-all-sorted-completions: Fix sorting performance bug * lisp/minibuffer.el (completion-all-sorted-completions): Use hash table for sorting by history position, O(m+n*log(n)) instead of O(m*n*log(n)) with history length `m` and candidate length `n`. --- diff --git a/lisp/minibuffer.el b/lisp/minibuffer.el index 06a5e1e988c..dde700fcf60 100644 --- a/lisp/minibuffer.el +++ b/lisp/minibuffer.el @@ -1393,14 +1393,31 @@ scroll the window of possible completions." (if (and (minibufferp) (not (eq minibuffer-history-variable t))) ;; Prefer recently used completions and put the default, if ;; it exists, on top. - (let ((hist (symbol-value minibuffer-history-variable))) - (setq all - (sort all - (lambda (c1 c2) - (cond ((equal c1 minibuffer-default) t) - ((equal c2 minibuffer-default) nil) - (t (> (length (member c1 hist)) - (length (member c2 hist)))))))))))) + (let* ((hist (symbol-value minibuffer-history-variable)) + (hash (make-hash-table :test #'equal :size (length hist))) + (index 0) + (def (car-safe minibuffer-default))) + ;; Record history positions in hash + (dolist (c hist) + (unless (gethash c hash) + (puthash c index hash)) + (cl-incf index)) + (when (stringp def) + (puthash def -1 hash)) + ;; Decorate elements with history position + (let ((c all)) + (while c + (setcar c (cons (gethash (car c) hash + most-positive-fixnum) + (car c))) + (pop c))) + (setq all (sort all #'car-less-than-car)) + ;; Drop decoration from the elements + (let ((c all)) + (while c + (setcar c (cdar c)) + (pop c))))))) + ;; Cache the result. This is not just for speed, but also so that ;; repeated calls to minibuffer-force-complete can cycle through ;; all possibilities.