]> git.eshelyaron.com Git - emacs.git/commitdiff
*** empty log message ***
authorDave Love <fx@gnu.org>
Mon, 3 Feb 2003 15:00:57 +0000 (15:00 +0000)
committerDave Love <fx@gnu.org>
Mon, 3 Feb 2003 15:00:57 +0000 (15:00 +0000)
etc/NEWS
lisp/ChangeLog
lisp/emacs-lisp/benchmark.el [new file with mode: 0644]

index dbf6cf99abee27517fae46664b02cb0766039b46..c8bdfdefb3b71ee14a88ce94c4707e054b0f070a 100644 (file)
--- a/etc/NEWS
+++ b/etc/NEWS
@@ -1144,6 +1144,9 @@ mode-lines in inverse-video.
 ** The obsolete C mode (c-mode.el) has been removed to avoid problems
 with Custom.
 
+** New package benchmark.el contains simple support for convenient
+timing measurements of code (including the garbage collection component).
+
 \f
 * Lisp Changes in Emacs 21.4
 
index 28cf0a9ae65277d111ddae513d46d32a7af2072f..3071991ba359b85783821b1031490af490d9c4e6 100644 (file)
@@ -1,3 +1,7 @@
+2003-02-03  Dave Love  <fx@gnu.org>
+
+       * emacs-lisp/benchmark.el: New.
+
 2003-02-02  Andreas Schwab  <schwab@suse.de>
 
        * progmodes/asm-mode.el (asm-font-lock-keywords): Allow arbitrary
diff --git a/lisp/emacs-lisp/benchmark.el b/lisp/emacs-lisp/benchmark.el
new file mode 100644 (file)
index 0000000..cc8a7bf
--- /dev/null
@@ -0,0 +1,117 @@
+;;; benchmark.el --- support for benchmarking code
+
+;; Copyright (C) 2003  Free Software Foundation, Inc.
+
+;; Author: Dave Love  <fx@gnu.org>
+;; Keywords: lisp, extensions
+
+;; This file 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 2, or (at your option)
+;; any later version.
+
+;; This file 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; see the file COPYING.  If not, write to
+;; the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+;; Boston, MA 02111-1307, USA.
+
+;;; Commentary:
+
+;; Utilities for timing the execution of forms, including the time
+;; taken for GC.  Note that prior to timing code you may want to
+;; ensure things like:  there has just been a GC, the relevant code is
+;; already loaded (so that there's no overhead from autoloading etc.),
+;; and the code is compiled if appropriate (but see
+;; `benchmark-run-compiled').
+
+;;; Code:
+
+(defmacro benchmark-elapse (&rest forms)
+  "Return the time in seconds elapsed for execution of FORMS."
+  (let ((t1 (make-symbol "t1"))
+       (t2 (make-symbol "t2")))
+    `(let (,t1 ,t2)
+       (setq ,t1 (current-time))
+       ,@forms
+       (setq ,t2 (current-time))
+       (+ (* (- (car ,t2) (car ,t1)) 65536.0)
+         (- (nth 1 ,t2) (nth 1 ,t1))
+         (* (- (nth 2 ,t2) (nth 2 ,t1)) 1.0e-6)))))
+(put 'benchmark-elapse 'edebug-form-spec t)
+(put 'benchmark-elapse 'lisp-indent-function 0)
+
+;;;###autoload
+(defmacro benchmark-run (&optional repetitions &rest forms)
+  "Time execution of FORMS.
+If REPETITIONS is supplied as a number, run forms that many times,
+accounting for the overhead of the resulting loop.  Otherwise run
+FORMS once.
+Return a list of the total elapsed time for execution, the number of
+garbage collections that ran, and the time taken by garbage collection.
+See also `benchmark-run-compiled'."
+  (unless (natnump repetitions)
+    (setq forms (cons repetitions forms)
+         repetitions 1))
+  (let ((i (make-symbol "i"))
+       (gcs (make-symbol "gcs"))
+       (gc (make-symbol "gc")))
+    `(let ((,gc gc-elapsed)
+          (,gcs gcs-done))
+       (list ,(if (> repetitions 1)
+                 ;; Take account of the loop overhead.
+                 `(- (benchmark-elapse (dotimes (,i ,repetitions)
+                                         ,@forms))
+                     (benchmark-elapse (dotimes (,i ,repetitions))))
+               `(benchmark-elapse ,@forms))
+            (- gcs-done ,gcs)
+            (- gc-elapsed ,gc)))))
+(put 'benchmark-run 'edebug-form-spec t)
+(put 'benchmark-run 'lisp-indent-function 2)
+
+;;;###autoload
+(defmacro benchmark-run-compiled (&optional repetitions &rest forms)
+  "Time execution of compiled version of FORMS.
+This is like `benchmark-run', but what is timed is a funcall of the
+byte code obtained by wrapping FORMS in a `lambda' and compiling the
+result.  The overhead of the `lambda's is accounted for."
+  (unless (natnump repetitions)
+    (setq forms (cons repetitions forms)
+         repetitions 1))
+  (let ((i (make-symbol "i"))
+       (gcs (make-symbol "gcs"))
+       (gc (make-symbol "gc"))
+       (code (byte-compile `(lambda () ,@forms)))
+       (lambda-code (byte-compile `(lambda ()))))
+    `(let ((,gc gc-elapsed)
+          (,gcs gcs-done))
+       (list ,(if (> repetitions 1)
+                 ;; Take account of the loop overhead.
+                 `(- (benchmark-elapse (dotimes (,i ,repetitions)
+                                         (funcall ,code)))
+                     (benchmark-elapse (dotimes (,i ,repetitions)
+                                         (funcall ,lambda-code))))
+               `(benchmark-elapse (funcall ,code)))
+            (- gcs-done ,gcs) (- gc-elapsed ,gc)))))
+(put 'benchmark-run-compiled 'edebug-form-spec t)
+(put 'benchmark-run-compiled 'lisp-indent-function 2)
+
+;;;###autoload
+(defun benchmark (repetitions form)
+  "Print the time taken for REPETITIONS executions of FORM.
+Interactively, REPETITIONS is taken from the prefix arg.  For
+non-interactive use see also `benchmark-run' and
+`benchmark-run-compiled'."
+  (interactive "p\nxForm: ")
+  (let ((result (eval `(benchmark-run ,repetitions ,form))))
+    (if (zerop (nth 1 result))
+       (message "Elapsed time: %fs" (car result))
+      (message "Elapsed time: %fs (%fs in %d GCs)" (car result)
+              (nth 2 result) (nth 1 result)))))
+
+(provide 'benchmark)
+;;; benchmark.el ends here