Mercurial > hg > xemacs-beta
view lisp/diagnose.el @ 5882:bbe4146603db
Reduce regexp usage, now CL-oriented non-regexp code available, core Lisp
lisp/ChangeLog addition:
2015-04-01 Aidan Kehoe <kehoea@parhasard.net>
When calling #'string-match with a REGEXP without regular
expression special characters, call #'search, #'mismatch, #'find,
etc. instead, making our code less likely to side-effect other
functions' match data and a little faster.
* apropos.el (apropos-command):
* apropos.el (apropos):
Call (position ?\n ...) rather than (string-match "\n" ...) here.
* buff-menu.el:
* buff-menu.el (buffers-menu-omit-invisible-buffers):
Don't fire up the regexp engine just to check if a string starts
with a space.
* buff-menu.el (select-buffers-tab-buffers-by-mode):
Don't fire up the regexp engine just to compare mode basenames.
* buff-menu.el (format-buffers-tab-line):
* buff-menu.el (build-buffers-tab-internal): Moved to being a
label within the following.
* buff-menu.el (buffers-tab-items): Use the label.
* bytecomp.el (byte-compile-log-1):
Don't fire up the regexp engine just to look for a newline.
* cus-edit.el (get):
Ditto.
* cus-edit.el (custom-variable-value-create):
Ditto, but for a colon.
* descr-text.el (describe-text-sexp):
Ditto.
* descr-text.el (describe-char-unicode-data):
Use #'split-string-by-char given that we're just looking for a
semicolon.
* descr-text.el (describe-char):
Don't fire up the regexp engine just to look for a newline.
* disass.el (disassemble-internal):
Ditto.
* files.el (file-name-sans-extension):
Implement this using #'position.
* files.el (file-name-extension):
Correct this function's docstring, implement it in terms of
#'position.
* files.el (insert-directory):
Don't fire up the regexp engine to split a string by space; don't
reverse the list of switches, this is actually a longstand bug as
far as I can see.
* gnuserv.el (gnuserv-process-filter):
Use #'position here, instead of consing inside #'split-string
needlessly.
* gtk-file-dialog.el (gtk-file-dialog-update-dropdown):
Use #'split-string-by-char here, don't fire up #'split-string for
directory-sep-char.
* gtk-font-menu.el (hack-font-truename):
Implement this more cheaply in terms of #'find,
#'split-string-by-char, #'equal, rather than #'string-match,
#'split-string, #'string-equal.
* hyper-apropos.el (hyper-apropos-grok-functions):
* hyper-apropos.el (hyper-apropos-grok-variables):
Look for a newline using #'position rather than #'string-match in
these functions.
* info.el (Info-insert-dir):
* info.el (Info-insert-file-contents):
* info.el (Info-follow-reference):
* info.el (Info-extract-menu-node-name):
* info.el (Info-menu):
Look for fixed strings using #'position or #'search as appropriate
in this file.
* ldap.el (ldap-decode-string):
* ldap.el (ldap-encode-string):
#'encode-coding-string, #'decode-coding-string are always
available, don't check if they're fboundp.
* ldap.el (ldap-decode-address):
* ldap.el (ldap-encode-address):
Use #'split-string-by-char in these functions.
* lisp-mnt.el (lm-creation-date):
* lisp-mnt.el (lm-last-modified-date):
Don't fire up the regexp engine just to look for spaces in this file.
* menubar-items.el (default-menubar):
Use (not (mismatch ...)) rather than #'string-match here, for
simple regexp.
Use (search "beta" ...) rather than (string-match "beta" ...)
* menubar-items.el (sort-buffers-menu-alphabetically):
* menubar-items.el (sort-buffers-menu-by-mode-then-alphabetically):
* menubar-items.el (group-buffers-menu-by-mode-then-alphabetically):
Don't fire up the regexp engine to check if a string starts with
a space or an asterisk.
Use the more fine-grained results of #'compare-strings; compare
case-insensitively for the buffer menu.
* menubar-items.el (list-all-buffers):
* menubar-items.el (tutorials-menu-filter):
Use #'equal rather than #'string-equal, which, in this context,
has the drawback of not having a bytecode, and no redeeming
features.
* minibuf.el:
* minibuf.el (un-substitute-in-file-name):
Use #'count, rather than counting the occurences of $ using the
regexp engine.
* minibuf.el (read-file-name-internal-1):
Don't fire up the regexp engine to search for ?=.
* mouse.el (mouse-eval-sexp):
Check for newline with #'find.
* msw-font-menu.el (mswindows-reset-device-font-menus):
Split a string by newline with #'split-string-by-char.
* mule/japanese.el:
* mule/japanese.el ("Japanese"):
Use #'search rather than #'string-match; canoncase before
comparing; fix a bug I had introduced where I had been making case
insensitive comparisons where the case mattered.
* mule/korea-util.el (default-korean-keyboard):
Look for ?3 using #'find, not #'string-march.
* mule/korea-util.el (quail-hangul-switch-hanja):
Search for a fixed string using #'search.
* mule/mule-cmds.el (set-locale-for-language-environment):
#'position, #'substitute rather than #'string-match,
#'replace-in-string.
* newcomment.el (comment-make-extra-lines):
Use #'search rather than #'string-match for a simple string.
* package-get.el (package-get-remote-filename):
Use #'position when looking for ?@
* process.el (setenv):
* process.el (read-envvar-name):
Use #'position when looking for ?=.
* replace.el (map-query-replace-regexp):
Use #'split-string-by-char instead of using an inline
implementation of it.
* select.el (select-convert-from-cf-text):
* select.el (select-convert-from-cf-unicodetext):
Use #'position rather than #'string-match in these functions.
* setup-paths.el (paths-emacs-data-root-p):
Use #'search when looking for simple string.
* sound.el (load-sound-file):
Use #'split-string-by-char rather than an inline reimplementation
of same.
* startup.el (splash-screen-window-body):
* startup.el (splash-screen-tty-body):
Search for simple strings using #'search.
* version.el (emacs-version):
Ditto.
* x-font-menu.el (hack-font-truename):
Implement this more cheaply in terms of #'find,
#'split-string-by-char, #'equal, rather than #'string-match,
#'split-string, #'string-equal.
* x-font-menu.el (x-reset-device-font-menus-core):
Use #'split-string-by-char here.
* x-init.el (x-initialize-keyboard):
Search for a simple string using #'search.
author | Aidan Kehoe <kehoea@parhasard.net> |
---|---|
date | Wed, 01 Apr 2015 14:28:20 +0100 |
parents | 3bc58dc9d688 |
children |
line wrap: on
line source
;;; diagnose.el --- routines for debugging problems in XEmacs ;; Copyright (C) 2002, 2010 Ben Wing. ;; Maintainer: XEmacs Development Team ;; Keywords: dumped ;; This file is part of XEmacs. ;; XEmacs 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, either version 3 of the License, or (at your ;; option) any later version. ;; XEmacs 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 XEmacs. If not, see <http://www.gnu.org/licenses/>. ;;; Synched up with: Not in FSF. ;;; Commentary: ;; This file is dumped with XEmacs. ;;; Code: (defun show-memory-usage () "Show statistics about memory usage of various sorts in XEmacs." (interactive) (garbage-collect) (labels ((show-foo-stats (objtypename statname-plist cleanfun objlist &optional objnamelen) (let* ((hash (make-hash-table)) (first t) types origtypes fmt (objnamelen (or objnamelen 25)) (linelen objnamelen) (totaltotal 0)) (loop for obj in objlist do (let ((total 0) (stats (object-memory-usage obj))) ;; Pop off the slice describing the object itself's ;; memory (while (and stats (not (eq t (pop stats))))) ;; Pop off the slice describing the associated ;; non-Lisp-Object memory from the allocation ;; perspective, so we can get to the slice describing ;; the memory grouped by type (while (and stats (pop stats))) (loop for (type . num) in (remq t stats) while type do (if first (push type origtypes)) (setq type (getf statname-plist type type)) (puthash type (+ num (or (gethash type hash) 0)) hash) (incf total num) (if first (push type types))) (incf totaltotal total) (when first (setq types (nreverse types)) (setq origtypes (nreverse origtypes)) (setq fmt (concat (format "%%-%ds" objnamelen) (mapconcat #'(lambda (type) (let ((fieldlen (max 7 (+ 2 (length (symbol-name type)))))) (incf linelen fieldlen) (format "%%%ds" fieldlen))) types "") (progn (incf linelen 9) "%9s\n"))) (princ "\n") (princ (apply 'format fmt objtypename (append types (list 'total)))) (princ (make-string linelen ?-)) (princ "\n")) (let ((objname (format "%s" (funcall cleanfun obj)))) (princ (apply 'format fmt (substring objname 0 (min (length objname) (1- objnamelen))) (nconc (mapcar #'(lambda (type) (cdr (assq type stats))) origtypes) (list total))))) (setq first nil))) (princ "\n") (princ (apply 'format fmt "total" (nconc (mapcar #'(lambda (type) (gethash type hash)) types) (list totaltotal)))) totaltotal))) (let ((grandtotal 0) (buffer "*memory usage*") begin) (with-output-to-temp-buffer buffer (save-excursion (set-buffer buffer) (when-fboundp 'charset-list (setq begin (point)) (incf grandtotal (show-foo-stats 'charset nil 'charset-name (mapcar 'get-charset (charset-list)))) (when-fboundp 'sort-numeric-fields (sort-numeric-fields -1 (save-excursion (goto-char begin) (forward-line 2) (point)) (save-excursion (forward-line -2) (point)))) (princ "\n")) (setq begin (point)) (incf grandtotal (show-foo-stats 'buffer nil 'buffer-name (buffer-list))) (when-fboundp 'sort-numeric-fields (sort-numeric-fields -1 (save-excursion (goto-char begin) (forward-line 3) (point)) (save-excursion (forward-line -2) (point)))) (princ "\n") (setq begin (point)) (incf grandtotal (show-foo-stats 'window '(line-start-cache line-st. face-cache face glyph-cache glyph redisplay-structs redisplay scrollbar-instances scrollbar window-mirror mirror) #'(lambda (x) (buffer-name (window-buffer x))) (mapcan #'(lambda (fr) (window-list fr t)) (frame-list)) 16)) (when-fboundp 'sort-numeric-fields (sort-numeric-fields -1 (save-excursion (goto-char begin) (forward-line 3) (point)) (save-excursion (forward-line -2) (point)))) (princ "\n") (let ((total 0) (fmt "%-30s%10s\n")) (setq begin (point)) (princ (format fmt "object" "storage")) (princ (make-string 40 ?-)) (princ "\n") (map-plist #'(lambda (stat num) (when (and (not (string-match "\\(.*\\)-ancillary-storage$" (symbol-name stat))) (string-match "\\(.*\\)-storage$" (symbol-name stat))) (incf total num) (princ (format fmt (match-string 1 (symbol-name stat)) num))) ) (sixth (garbage-collect))) (princ "\n") (princ (format fmt "total" total)) (incf grandtotal total)) (when-fboundp 'sort-numeric-fields (sort-numeric-fields -1 (save-excursion (goto-char begin) (forward-line 2) (point)) (save-excursion (forward-line -2) (point)))) (princ (format "\n\ngrand total: %s\n" grandtotal))) grandtotal)))) (defun show-object-memory-usage-stats () "Show statistics about object memory usage in XEmacs." (interactive) (garbage-collect) (let ((buffer "*object memory usage statistics*") (plist (object-memory-usage-stats)) (fmt "%-28s%10s%10s%10s%10s%10s\n") (grandtotal 0) begin) (labels ((show-stats (match-string) (princ (format "%28s%10s%40s\n" "" "" "--------------storage---------------")) (princ (format fmt "object" "count" "object" "overhead" "non-Lisp" "ancillary")) (princ (make-string 78 ?-)) (princ "\n") (let ((total-use 0) (total-non-lisp-use 0) (total-use-overhead 0) (total-use-with-overhead 0) (total-count 0)) (map-plist #'(lambda (stat num) (let ((symmatch (and (string-match match-string (symbol-name stat)) (match-string 1 (symbol-name stat))))) (when (and symmatch (or (< (length symmatch) 9) (not (equal (substring symmatch -9) "-non-lisp"))) (or (< (length symmatch) 15) (not (equal (substring symmatch -15) "-lisp-ancillary")))) (let* ((storage-use num) (storage-use-overhead (or (plist-get plist (intern (concat symmatch "-storage-overhead"))) 0)) (storage-use-with-overhead (or (plist-get plist (intern (concat symmatch "-storage-including-overhead"))) (+ storage-use storage-use-overhead))) (storage-use-overhead (- storage-use-with-overhead storage-use)) (non-lisp-storage (or (plist-get plist (intern (concat symmatch "-non-lisp-storage"))) 0)) (lisp-ancillary-storage (or (plist-get plist (intern (concat symmatch "-lisp-ancillary-storage"))) 0)) (storage-count (or (loop for str in '("s-used" "es-used" "-used") for val = (plist-get plist (intern (concat symmatch str))) if val return val) (plist-get plist (intern (concat (substring symmatch 0 -1) "ies-used"))) ))) (incf total-use storage-use) (incf total-use-overhead storage-use-overhead) (incf total-use-with-overhead storage-use-with-overhead) (incf total-non-lisp-use non-lisp-storage) (incf total-count (or storage-count 0)) (and (> storage-use-with-overhead 0) (princ (format fmt symmatch (or storage-count "unknown") storage-use storage-use-overhead non-lisp-storage lisp-ancillary-storage))))))) plist) (princ "\n") (princ (format fmt "total" total-count total-use total-use-overhead total-non-lisp-use "")) (incf grandtotal total-use-with-overhead) (incf grandtotal total-non-lisp-use) (when-fboundp 'sort-numeric-fields (sort-numeric-fields -4 (save-excursion (goto-char begin) (forward-line 4) (point)) (save-excursion (forward-line -2) (point))))))) (with-output-to-temp-buffer buffer (save-excursion (set-buffer buffer) (setq begin (point)) (princ "Allocated with lisp allocator or related:\n") (show-stats "\\(.*\\)-storage$") (princ (format "\n\ngrand total: %s\n" grandtotal))) grandtotal)))) (defun show-mc-alloc-memory-usage () "Show statistics about memory usage of the new allocator." (interactive) (garbage-collect) (if-fboundp 'mc-alloc-memory-usage (let* ((stats (mc-alloc-memory-usage)) (page-size (first stats)) (heap-sects (second stats)) (used-plhs (third stats)) (free-plhs (fourth stats)) (globals (fifth stats)) (mc-malloced-bytes (sixth stats))) (with-output-to-temp-buffer "*mc-alloc memory usage*" (labels ((print-used-plhs (text plhs) (let ((sum-n-pages 0) (sum-used-n-cells 0) (sum-used-space 0) (sum-used-total 0) (sum-total-n-cells 0) (sum-total-space 0) (sum-total-total 0) (fmt "%7s%7s|%7s%9s%9s%4s|%7s%9s%9s%4s|%4s\n")) (princ (format "%-14s|%-29s|%-29s|\n" text " currently in use" " total available")) (princ (format fmt "cell-sz" "#pages" "#cells" "space" "total" "% " "#cells" "space" "total" "% " "% ")) (princ (make-string 79 ?-)) (princ "\n") (while plhs (let* ((elem (car plhs)) (cell-size (first elem)) (n-pages (second elem)) (used-n-cells (third elem)) (used-space (fourth elem)) (used-total (if (zerop cell-size) (sixth elem) (* cell-size used-n-cells))) (used-eff (floor (if (not (zerop used-total)) (* (/ (* used-space 1.0) (* used-total 1.0)) 100.0) 0))) (total-n-cells (fifth elem)) (total-space (if (zerop cell-size) used-space (* cell-size total-n-cells))) (total-total (sixth elem)) (total-eff (floor (if (not (zerop total-total)) (* (/ (* total-space 1.0) (* total-total 1.0)) 100.0) 0))) (eff (floor (if (not (zerop total-total)) (* (/ (* used-space 1.0) (* total-total 1.0)) 100.0) 0)))) (princ (format fmt cell-size n-pages used-n-cells used-space used-total used-eff total-n-cells total-space total-total total-eff eff)) (incf sum-n-pages n-pages) (incf sum-used-n-cells used-n-cells) (incf sum-used-space used-space) (incf sum-used-total used-total) (incf sum-total-n-cells total-n-cells) (incf sum-total-space total-space) (incf sum-total-total total-total)) (setq plhs (cdr plhs))) (let ((avg-used-eff (floor (if (not (zerop sum-used-total)) (* (/ (* sum-used-space 1.0) (* sum-used-total 1.0)) 100.0) 0))) (avg-total-eff (floor (if (not (zerop sum-total-total)) (* (/ (* sum-total-space 1.0) (* sum-total-total 1.0)) 100.0) 0))) (avg-eff (floor (if (not (zerop sum-total-total)) (* (/ (* sum-used-space 1.0) (* sum-total-total 1.0)) 100.0) 0)))) (princ (format fmt "sum " sum-n-pages sum-used-n-cells sum-used-space sum-used-total avg-used-eff sum-total-n-cells sum-total-space sum-total-total avg-total-eff avg-eff)) (princ "\n")))) (print-free-plhs (text plhs) (let ((sum-n-pages 0) (sum-n-sects 0) (sum-space 0) (sum-total 0) (fmt "%6s%10s |%7s%10s\n")) (princ (format "%s\n" text)) (princ (format fmt "#pages" "space" "#sects" "total")) (princ (make-string 35 ?-)) (princ "\n") (while plhs (let* ((elem (car plhs)) (n-pages (first elem)) (n-sects (second elem)) (space (* n-pages page-size)) (total (* n-sects space))) (princ (format fmt n-pages space n-sects total)) (incf sum-n-pages n-pages) (incf sum-n-sects n-sects) (incf sum-space space) (incf sum-total total)) (setq plhs (cdr plhs))) (princ (make-string 35 ?=)) (princ "\n") (princ (format fmt sum-n-pages sum-space sum-n-sects sum-total)) (princ "\n")))) (princ (format "%-12s%10s\n\n" "PAGE_SIZE" page-size)) (print-used-plhs "USED HEAP" used-plhs) (princ "\n\n") (print-free-plhs "FREE HEAP" free-plhs) (princ "\n\n") (let ((fmt "%-30s%10s\n")) (princ (format fmt "heap sections" "")) (princ (make-string 40 ?-)) (princ "\n") (princ (format fmt "number of heap sects" (first heap-sects))) (princ (format fmt "used size" (second heap-sects))) (princ (make-string 40 ?-)) (princ "\n") (princ (format fmt "real size" (third heap-sects))) (princ (format fmt "global allocator structs" globals)) (princ (make-string 40 ?-)) (princ "\n") (princ (format fmt "real size + structs" (+ (third heap-sects) globals))) (princ "\n") (princ (make-string 40 ?=)) (princ "\n") (princ (format fmt "grand total" mc-malloced-bytes))) (+ mc-malloced-bytes)))) (message "mc-alloc not used in this XEmacs."))) (defun show-gc-stats () "Show statistics about garbage collection cycles." (interactive) (if-fboundp 'gc-stats (let ((buffer "*garbage collection statistics*") (plist (gc-stats)) (fmt "%-9s %16s %12s %12s %12s %12s\n")) (labels ((plist-get-stat (category field) (let ((stat (plist-get plist (intern (concat category field))))) (if stat (format "%.0f" stat) "-"))) (show-stats (category) (princ (format fmt category (plist-get-stat category "-total") (plist-get-stat category "-in-last-gc") (plist-get-stat category "-in-this-gc") (plist-get-stat category "-in-last-cycle") (plist-get-stat category "-in-this-cycle"))))) (with-output-to-temp-buffer buffer (save-excursion (set-buffer buffer) (princ (format "%s %g\n" "Current phase" (plist-get plist 'phase))) (princ (make-string 78 ?-)) (princ "\n") (princ (format fmt "stat" "total" "last-gc" "this-gc" "last-cycle" "this-cycle")) (princ (make-string 78 ?-)) (princ "\n") (show-stats "n-gc") (show-stats "n-cycles") (show-stats "enqueued") (show-stats "dequeued") (show-stats "repushed") (show-stats "enqueued2") (show-stats "dequeued2") (show-stats "finalized") (show-stats "freed") (plist-get plist 'n-gc-total))))) (error 'void-function "gc-stats not available.")))