[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
&Rest Lists
- To: Common-Lisp@SAIL.STANFORD.EDU
- Subject: &Rest Lists
- From: ELIOT@cs.umass.edu
- Date: Fri, 1 Apr 88 15:15 EDT
In order to clarify the importance of optimized handling of &Rest
arguments, compared to the importance of specifying their semantics more
precisely I have collected some statistics. The functions used to
collect these are included at the end of this message. I encourage
others to use these functions to obtain measurements of their own
code.
Using APROPOS it is possible to measure how many functions take &Rest
arguments. On an Explorer I got the following results:
PACKAGE with &Rest Without Percent
------------------------------------
LISP 70 580 10.7
SI 73 3384 2.1
ZWEI 66 2324 2.7
TV 79 1177 6.2
FS 54 707 7.0
FORMAT 11 105 9.4
------------------------------------
Subtotal 283 7697 3.5
KEE 234 5120 4.3 (IntelliCorp's package)
------------------------------------
Total 517 12817 3.8
Unfortunately, since calls to APPLY are optimized they cannot be
analyzed with WHO-CALLS. Instead I searched the source code of the NIL
compiler for lines containing the strings "(APPLY" and "&REST". These
include comment lines, and parsing of argument lists. (The paren in
"(APPLY" eliminates some of that.)
From nil$disk:[nil.h]*.lsp
Lines Textually Containing:
"(APPLY" "&Rest" "defun" TOTAL LINES
-------------------------------------------------
18 61 859 24799
So 7.1% of these function definitions take &rest arguments. (Including
NIL specific &Restv, &Restl.)
Too many assumptions must be made for a realistic estimate of
the actual optimization value of specially handling &rest
lists to be made using these numbers. These numbers all
refer to source code statistics, while the actual optimization
depends upon instruction stream characteristics. However,
the fact that &Rest is used in a small minority of function
definitions, while APPLY is very rare strongly suggests that
their combined use in a single function call is very rare
indeed. Consequently these statistics do not support the idea
that this case needs to be highly optimized.
The 70 Explorer LISP package (i.e. Common Lisp) functions with &Rest
arguments are:
------------------------------------------------------
;;; Unclassified:
MAPHASH MAKE-CONCATENATED-STREAM ARRAY-ROW-MAJOR-INDEX
MAKE-BROADCAST-STREAM ADJUST-ARRAY ROOM
;;; IO Functions:
YES-OR-NO-P CERROR ERROR FORMAT WARN BREAK Y-OR-N-P
;;; Primitives perhaps known to the compiler:
APPEND MAPC = MAX NOTANY SBIT BIT < MAPCON /= MAPLIST LIST* MAPCAN AREF
ARRAY-IN-BOUNDS-P / CONCATENATE MAPCAR MAPL - VECTOR FUNCALL MIN XOR
EVERY NOTEVERY <= APPLY + MAP VALUES * >= SOME LIST NCONC IGNORE >
LOGIOR LOGEQV LOGXOR LOGAND LCM GCD BOOLE
CHAR-NOT-GREATERP CHAR-NOT-LESSP CHAR-NOT-EQUAL CHAR> CHAR< CHAR/=
CHAR-EQUAL CHAR= CHAR<= CHAR-LESSP CHAR-GREATERP CHAR>=
------------------------------------------------------
The overall importance of optimizing CONSing also depends upon the speed
of CONSing. Since one of the most important known uses of APPLY where
&Rest arguments are an issue involves calling FORMAT it is important to
compare CONSing speed with I/O speed. On an Explorer II, GC-ON,
everything compiled:
Function Calls Time Normalized to 1,000,000 Calls
-------------------------------------------------------------------
(foo (cons 1 2)) 10,000,000 140.8 14.8
NIL 1,000,000 1.3 1.3
(foo 2) 1,000,000 4.53 4.53
(write-char #\a) 10,000 3.74 374.0
(probe-file ...) 100 43.7 437,000
[using a string as a pathname, non-local file,
connection already established.]
FOO is a null function needed to prevent optimization.
(dotimes (i 10000000) (cons 1 2)) is optimized to:
(dotimes (i 10000000) nil) so:
(dotimes (i 10000000) (foo (cons 1 2))) must be used
instead. FOO is defined with:
(defun foo (x) x)
On the basis of this I conclude that the common idiom
(apply #'format ...) does not deserve special attention
to the number of CONSes performed in creating the &Rest list
for FORMAT.
=================================================================
The actual functions used to collect some of this data follow.
;;; -*- Mode:Common-Lisp; Package:USER; Base:10 -*-
(defvar *count* 0)
(defvar *not-count* 0)
(defvar *fns* nil)
(defun rest-p (x)
(cond ((and (not (macro-function x))
(not (special-form-p x))
(member '&rest (arglist x)))
(push x *fns*)
(incf *count*) t)
(t (incf *not-count*)
nil)))
(defun test (pkg)
(setq *count* 0)
(setq *not-count* 0)
(setq *fns* nil)
(apropos "" :package pkg :inherited nil :predicate #'rest-p :fboundp t :inheritors nil)
(format t "~&There are ~a functions with &Rest arguments, ~a without in package ~a"
*count* *not-count* pkg))
(defun file-search (file str)
(let ((count 0)
(lines 0))
(with-open-file (stream file)
(loop for line = (read-line stream nil nil)
while line
do (incf lines)
(when (search str line :test #'char-equal)
(format t "~&~a" line)
(incf count))))
(values count lines)))
(defun file-search-list (files str)
(let ((count 0)
(lines 0))
(dolist (file files)
(multiple-value-bind (a b) (file-search file str)
(setq count (+ count a))
(setq lines (+ lines b))
(format t "~&File ~a Count ~a; lines ~a Total count ~a, lines ~a" file a b count lines)))
(format t "~&Grand Total count ~a; lines ~a" count lines)
(values count lines)))
;;; (file-search-list (pathname "nil$disk:[nil.h]*.lsp") "(APPLY")