mirror of
https://git.savannah.gnu.org/git/emacs.git
synced 2025-01-07 15:21:46 +00:00
330 lines
13 KiB
EmacsLisp
330 lines
13 KiB
EmacsLisp
;;; memory-report.el --- Short function summaries -*- lexical-binding: t -*-
|
|
|
|
;; Copyright (C) 2020-2023 Free Software Foundation, Inc.
|
|
|
|
;; Keywords: lisp, help
|
|
|
|
;; This file is part of GNU Emacs.
|
|
|
|
;; GNU Emacs 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.
|
|
|
|
;; GNU Emacs 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 GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
;;; Commentary:
|
|
|
|
;; Todo (possibly): Font cache, regexp cache, bidi cache, various
|
|
;; buffer caches (newline cache, free_region_cache, etc), composition
|
|
;; cache, face cache.
|
|
|
|
;;; Code:
|
|
|
|
(require 'seq)
|
|
(require 'subr-x)
|
|
(require 'cl-lib)
|
|
|
|
(defvar memory-report--type-size nil)
|
|
|
|
;;;###autoload
|
|
(defun memory-report ()
|
|
"Generate a report of how Emacs is using memory.
|
|
|
|
This report is approximate, and will commonly over-count memory
|
|
usage by variables, because shared data structures will usually
|
|
by counted more than once."
|
|
(interactive)
|
|
(pop-to-buffer "*Memory Report*")
|
|
(special-mode)
|
|
(button-mode 1)
|
|
(setq-local revert-buffer-function (lambda (_ignore-auto _noconfirm)
|
|
(memory-report)))
|
|
(setq truncate-lines t)
|
|
(message "Gathering data...")
|
|
(let ((reports (append (memory-report--garbage-collect)
|
|
(memory-report--image-cache)
|
|
(memory-report--symbol-plist)
|
|
(memory-report--buffers)
|
|
(memory-report--largest-variables)))
|
|
(inhibit-read-only t)
|
|
summaries details)
|
|
(message "Gathering data...done")
|
|
(erase-buffer)
|
|
(insert (propertize "Estimated Emacs Memory Usage\n\n" 'face 'bold))
|
|
(dolist (report reports)
|
|
(if (listp report)
|
|
(push report summaries)
|
|
(push report details)))
|
|
(dolist (summary (seq-sort (lambda (e1 e2)
|
|
(> (cdr e1) (cdr e2)))
|
|
summaries))
|
|
(insert (format "%s %s\n"
|
|
(memory-report--format (cdr summary))
|
|
(car summary))))
|
|
(insert "\n")
|
|
(dolist (detail (nreverse details))
|
|
(insert detail "\n")))
|
|
(goto-char (point-min)))
|
|
|
|
(defun memory-report-object-size (object)
|
|
"Return the size of OBJECT in bytes."
|
|
(unless memory-report--type-size
|
|
(memory-report--garbage-collect))
|
|
(memory-report--object-size (make-hash-table :test #'eq) object))
|
|
|
|
(defun memory-report--size (type)
|
|
(or (gethash type memory-report--type-size)
|
|
(gethash 'object memory-report--type-size)))
|
|
|
|
(defun memory-report--set-size (elems)
|
|
(setq memory-report--type-size (make-hash-table))
|
|
(setf (gethash 'string memory-report--type-size)
|
|
(cadr (assq 'strings elems)))
|
|
(setf (gethash 'cons memory-report--type-size)
|
|
(cadr (assq 'conses elems)))
|
|
(setf (gethash 'symbol memory-report--type-size)
|
|
(cadr (assq 'symbols elems)))
|
|
(setf (gethash 'object memory-report--type-size)
|
|
(cadr (assq 'vectors elems)))
|
|
(setf (gethash 'float memory-report--type-size)
|
|
(cadr (assq 'floats elems)))
|
|
(setf (gethash 'buffer memory-report--type-size)
|
|
(cadr (assq 'buffers elems))))
|
|
|
|
(defun memory-report--garbage-collect ()
|
|
(let ((elems (garbage-collect)))
|
|
(memory-report--set-size elems)
|
|
(let ((data (list
|
|
(list 'strings
|
|
(+ (memory-report--gc-elem elems 'strings)
|
|
(memory-report--gc-elem elems 'string-bytes)))
|
|
(list 'vectors
|
|
(+ (memory-report--gc-elem elems 'vectors)
|
|
(memory-report--gc-elem elems 'vector-slots)))
|
|
(list 'floats (memory-report--gc-elem elems 'floats))
|
|
(list 'conses (memory-report--gc-elem elems 'conses))
|
|
(list 'symbols (memory-report--gc-elem elems 'symbols))
|
|
(list 'intervals (memory-report--gc-elem elems 'intervals))
|
|
(list 'buffer-objects
|
|
(memory-report--gc-elem elems 'buffers)))))
|
|
(list (cons "Overall Object Memory Usage"
|
|
(seq-reduce #'+ (mapcar (lambda (elem)
|
|
(* (nth 1 elem) (nth 2 elem)))
|
|
elems)
|
|
0))
|
|
(cons "Reserved (But Unused) Object Memory"
|
|
(seq-reduce #'+ (mapcar (lambda (elem)
|
|
(if (nth 3 elem)
|
|
(* (nth 1 elem) (nth 3 elem))
|
|
0))
|
|
elems)
|
|
0))
|
|
(with-temp-buffer
|
|
(insert (propertize "Object Storage\n\n" 'face 'bold))
|
|
(dolist (object (seq-sort (lambda (e1 e2)
|
|
(> (cadr e1) (cadr e2)))
|
|
data))
|
|
(insert (format "%s %s\n"
|
|
(memory-report--format (cadr object))
|
|
(capitalize (symbol-name (car object))))))
|
|
(buffer-string))))))
|
|
|
|
(defun memory-report--largest-variables ()
|
|
(let ((variables nil))
|
|
(mapatoms
|
|
(lambda (symbol)
|
|
(when (boundp symbol)
|
|
(let ((size (memory-report--object-size
|
|
(make-hash-table :test #'eq)
|
|
(symbol-value symbol))))
|
|
(when (> size 1000)
|
|
(push (cons symbol size) variables)))))
|
|
obarray)
|
|
(list
|
|
(cons (propertize "Memory Used By Global Variables"
|
|
'help-echo "Upper bound; mutually overlapping data from different variables are counted several times")
|
|
(seq-reduce #'+ (mapcar #'cdr variables) 0))
|
|
(with-temp-buffer
|
|
(insert (propertize "Largest Variables\n\n" 'face 'bold))
|
|
(cl-loop for i from 1 upto 20
|
|
for (symbol . size) in (seq-sort (lambda (e1 e2)
|
|
(> (cdr e1) (cdr e2)))
|
|
variables)
|
|
do (insert (memory-report--format size)
|
|
" "
|
|
(symbol-name symbol)
|
|
"\n"))
|
|
(buffer-string)))))
|
|
|
|
(defun memory-report--symbol-plist ()
|
|
(let ((counted (make-hash-table :test #'eq))
|
|
(total 0))
|
|
(mapatoms
|
|
(lambda (symbol)
|
|
(cl-incf total (memory-report--object-size
|
|
counted (symbol-plist symbol))))
|
|
obarray)
|
|
(list
|
|
(cons "Memory Used By Symbol Plists" total))))
|
|
|
|
(defun memory-report--object-size (counted value)
|
|
(if (gethash value counted)
|
|
0
|
|
(setf (gethash value counted) t)
|
|
(memory-report--object-size-1 counted value)))
|
|
|
|
(cl-defgeneric memory-report--object-size-1 (_counted _value)
|
|
0)
|
|
|
|
;; This shouldn't happen, but there's some leakage.
|
|
(cl-defmethod memory-report--object-size-1 (_ (_value symbol-with-pos))
|
|
(memory-report--size 'symbol))
|
|
|
|
(cl-defmethod memory-report--object-size-1 (_ (value symbol))
|
|
;; Don't count global symbols -- makes sizes of lists of symbols too
|
|
;; heavy.
|
|
(if (intern-soft value obarray)
|
|
0
|
|
(memory-report--size 'symbol)))
|
|
|
|
(cl-defmethod memory-report--object-size-1 (_ (_value buffer))
|
|
(memory-report--size 'buffer))
|
|
|
|
(cl-defmethod memory-report--object-size-1 (counted (value string))
|
|
(+ (memory-report--size 'string)
|
|
(string-bytes value)
|
|
(memory-report--interval-size counted (object-intervals value))))
|
|
|
|
(defun memory-report--interval-size (counted intervals)
|
|
;; We get a list back of intervals, but only count the "inner list"
|
|
;; (i.e., the actual text properties), and add the size of the
|
|
;; intervals themselves.
|
|
(+ (* (memory-report--size 'interval) (length intervals))
|
|
(seq-reduce #'+ (mapcar
|
|
(lambda (interval)
|
|
(memory-report--object-size counted (nth 2 interval)))
|
|
intervals)
|
|
0)))
|
|
|
|
(cl-defmethod memory-report--object-size-1 (counted (value list))
|
|
(let ((total 0)
|
|
(size (memory-report--size 'cons)))
|
|
(while value
|
|
(cl-incf total size)
|
|
(setf (gethash value counted) t)
|
|
(when (car value)
|
|
(cl-incf total (memory-report--object-size counted (car value))))
|
|
(let ((next (cdr value)))
|
|
(setq value (when next
|
|
(if (consp next)
|
|
(unless (gethash next counted)
|
|
(cdr value))
|
|
(cl-incf total (memory-report--object-size
|
|
counted next))
|
|
nil)))))
|
|
total))
|
|
|
|
(cl-defmethod memory-report--object-size-1 (counted (value vector))
|
|
(let ((total (+ (memory-report--size 'vector)
|
|
(* (memory-report--size 'object) (length value)))))
|
|
(cl-loop for elem across value
|
|
do (cl-incf total (memory-report--object-size counted elem)))
|
|
total))
|
|
|
|
(cl-defmethod memory-report--object-size-1 (counted (value hash-table))
|
|
(let ((total (+ (memory-report--size 'vector)
|
|
(* (memory-report--size 'object) (hash-table-size value)))))
|
|
(maphash
|
|
(lambda (key elem)
|
|
(cl-incf total (memory-report--object-size counted key))
|
|
(cl-incf total (memory-report--object-size counted elem)))
|
|
value)
|
|
total))
|
|
|
|
;; All cl-defstruct types.
|
|
(cl-defmethod memory-report--object-size-1 (counted (value cl-structure-object))
|
|
(let ((struct-type (type-of value)))
|
|
(apply #'+
|
|
(memory-report--size 'vector)
|
|
(mapcar (lambda (slot)
|
|
(if (eq (car slot) 'cl-tag-slot)
|
|
0
|
|
(memory-report--object-size
|
|
counted
|
|
(cl-struct-slot-value struct-type (car slot) value))))
|
|
(cl-struct-slot-info struct-type)))))
|
|
|
|
(defun memory-report--format (bytes)
|
|
(format "%10s" (file-size-human-readable bytes 'iec " ")))
|
|
|
|
(defun memory-report--gc-elem (elems type)
|
|
(* (nth 1 (assq type elems))
|
|
(nth 2 (assq type elems))))
|
|
|
|
(defun memory-report--buffers ()
|
|
(let ((buffers (mapcar (lambda (buffer)
|
|
(cons buffer (memory-report--buffer buffer)))
|
|
(buffer-list))))
|
|
(list (cons "Total Buffer Memory Usage"
|
|
(seq-reduce #'+ (mapcar #'cdr buffers) 0))
|
|
(with-temp-buffer
|
|
(insert (propertize "Largest Buffers\n\n" 'face 'bold))
|
|
(cl-loop for i from 1 upto 20
|
|
for (buffer . size) in (seq-sort (lambda (e1 e2)
|
|
(> (cdr e1) (cdr e2)))
|
|
buffers)
|
|
do (insert (memory-report--format size)
|
|
" "
|
|
(buttonize
|
|
(buffer-name buffer)
|
|
#'memory-report--buffer-details buffer)
|
|
"\n"))
|
|
(buffer-string)))))
|
|
|
|
(defun memory-report--buffer-details (buffer)
|
|
(with-current-buffer buffer
|
|
(apply
|
|
#'message
|
|
"Buffer text: %s; variables: %s; text properties: %s; overlays: %s"
|
|
(mapcar #'string-trim (mapcar #'memory-report--format
|
|
(memory-report--buffer-data buffer))))))
|
|
|
|
(defun memory-report--buffer (buffer)
|
|
(seq-reduce #'+ (memory-report--buffer-data buffer) 0))
|
|
|
|
(defun memory-report--buffer-data (buffer)
|
|
(with-current-buffer buffer
|
|
(list (save-restriction
|
|
(widen)
|
|
(+ (position-bytes (point-max))
|
|
(- (position-bytes (point-min)))
|
|
(gap-size)))
|
|
(seq-reduce #'+ (mapcar (lambda (elem)
|
|
(if (and (consp elem) (cdr elem))
|
|
(memory-report--object-size
|
|
(make-hash-table :test #'eq)
|
|
(cdr elem))
|
|
0))
|
|
(buffer-local-variables buffer))
|
|
0)
|
|
(memory-report--object-size (make-hash-table :test #'eq)
|
|
(object-intervals buffer))
|
|
(memory-report--object-size (make-hash-table :test #'eq)
|
|
(overlay-lists)))))
|
|
|
|
(defun memory-report--image-cache ()
|
|
(list (cons "Total Image Cache Size" (if (fboundp 'image-cache-size)
|
|
(image-cache-size)
|
|
0))))
|
|
|
|
(provide 'memory-report)
|
|
|
|
;;; memory-report.el ends here
|